Решение систем линейных уравнений с помощью Python

Kate

Administrator
Команда форума

Немного про систему линейных уравнений​

Уравнение называется линейным, если оно содержит неизвестные только в первой степени и не содержит произведений неизвестных, т.е. если оно имеет вид
a_1x_1+a_2x_2+...+a_1n_n=b
, где
a_{ij}
- коэффициент уравнения, а
b
- константа, не зависящая от x.

Система линейных уравнений объединяет n таких уравнений, каждое из которых содержит k переменных.

Визуально систему линейных уравнений можно представить следующим образом:

\begin{equation*} \begin{cases}    a_{11}x_1 + a_{12}x_2 + ... + a_{1n}x_n = b_1,\\    a_{21}x_1 + a_{22}x_2 + ... + a_{2n}x_n = b_2,  \\ a_{m1}x_1 + a_{m2}x_2 + ... + a_{mn}x_n=b_m.  \end{cases} \end{equation*}

Системы линейных уравнений могут быть представлены в матричной форме
AX=B
, где
A
- это матрица коэффициентов системы линейных уравнений,
X
- вектор-столбец неизвестных, а
B
- вектор-столбец свободных членов. Библиотека SymPy предлагает функционал для работы с матрицами и позволяет решать системы уравнений через матричные операции.

Например, у нас есть система уравнений:

\begin{equation*} \begin{cases}    x_1 + x_2 + x_3-x_4 = 0,\\   5x_1 - 3x_2 +2 x_3-8x_4 = 1,  \\ 3x_1 + 5x_2 + x_3+4x_4 = 0, \\ 4x_1 + 2x_2 + 3x_3+x_4 = 3.  \end{cases} \end{equation*}
A = \begin{bmatrix} 1&1&1&1 \\ 5&-3&2&-8\\3&5&1&4\\4&2&3&1\end{bmatrix}, B=\begin{bmatrix}0\\1\\0\\3\end{bmatrix}, X=\begin{bmatrix}x_1\\x_2\\x_3\\x_4\\\end{bmatrix}

Предлагаю немного поиграть с ней и библиотекой.

Метод Крамера​

Правило Крамера представляет собой ключевой метод для нахождения решений системы линейных уравнений. Этот метод опирается на вычисление определителей соответствующих матриц, поэтому его часто называют методом определителей. В контексте системы уравнений
AX=B
, где
A
- матрица коэффициентов, а
B
- столбец значений, правило Крамера выражает решение в виде отношения определителей, включающих матрицы, полученные заменой столбцов
A
на столбец
B
. Применять правило Крамера можно только в случае, если определитель матрицы
A
отличен от нуля, что указывает на её обратимость.

from sympy import symbols, Matrix


# Функция применения метода Крамера для решения системы линейных уравнений
def cramer_rule(A, B):
# Вычисление определителя главной матрицы
det_A = A.det()
# Проверка на случай, если определитель главной матрицы равен нулю
if det_A == 0:
raise ValueError("Определитель матрицы коэффициентов равен нулю, метод Крамера не применим.")

solutions = []
# Проходим по каждому столбцу матрицы и вычисляем определитель со заменой столбца на вектор значений
for i in range(A.shape[0]):
Ai = A.copy()
Ai[:, i] = B
solutions.append(Ai.det() / det_A)

return solutions


# Объявление символьных переменных
x1, x2, x3, x4 = symbols('x1 x2 x3 x4')
# Матрица коэффициентов системы уравнений
A = Matrix([
[1, 1, 1, 1],
[5, -3, 2, -8],
[3, 5, 1, 4],
[4, 2, 3, 1]
])
# Вектор значений
B = Matrix([0, 1, 0, 3])

try:
# Вызов функции для решения системы уравнений методом Крамера
solutions = cramer_rule(A, B)
# Вывод результатов
print("Решение методом Крамера:")
for i, sol in enumerate(solutions, start=1):
print(f"x{i}: {sol}")
except ValueError as e:
# Вывод сообщения об ошибке, если определитель главной матрицы равен нулю
print(e)

ac94e16616ee0cac6391eb8580e8e467.png

В коде представлена функция под названием cramer_rule, предназначенная для применения правила Крамера при решении систем линейных уравнений. Она сначала рассчитывает определитель основной матрицы
A
, затем создаёт альтернативные матрицы, подменяя каждый из столбцов матрицы
A
вектором констант
B
вычисляя их определители. Решения переменных
x_i
получаются путём деления определителей этих альтернативных матриц на определитель основной матрицы
A
.

Затем в коде происходит объявление символьных переменных, а также формирование матрицы коэффициентов
A
и вектора-столбца
B
, что необходимо для проверки эффективности правила Крамера.

В рамках блока try-except осуществляется вызов функции cramer_rule, чтобы найти решение системы уравнений с использованием метода Крамера. В случае, когда определитель основной матрицы оказывается равным нулю, происходит возбуждение исключения ValueError с последующим отображением соответствующего сообщения о невозможности решения. Если же определитель не равен нулю, результаты, представляющие собой решения системы, выводятся на экран.

Метод Гаусса​

Метод Гаусса, известный также как гауссово исключение или преобразование системы уравнений к ступенчатой форме, представляет собой один из фундаментальных подходов к решению систем линейных уравнений. Этот метод включает выполнение элементарных операций над строками матрицы уравнений для её трансформации в ступенчатый или упрощённый ступенчатый вид, что значительно облегчает нахождение решений системы.

from sympy import Matrix, pprint


def print_row_reduced_matrix(matrix):
print("Ступенчатая матрица:")
pprint(matrix)


# Определение расширенной матрицы [A|B]
augmented_matrix = Matrix([
[1, 1, 1, 1, 0],
[5, -3, 2, -8, 1],
[3, 5, 1, 4, 0],
[4, 2, 3, 1, 3]
])

# Выполнение приведения матрицы к ступенчатому виду
row_reduced_matrix, _ = augmented_matrix.rref()

# Вызов функции для вывода ступенчатой матрицы
print_row_reduced_matrix(row_reduced_matrix)
3f821f0a2fe6a029753f8bc8592e894d.png

В представленном коде начинаем с определения расширенной матрицы, где в последнем столбце размещены свободные члены системы уравнений. Далее, используя метод .rref(), который предоставляется объектом Matrix библиотеки SymPy, преобразуем эту матрицу к ступенчатому виду.

После преобразования ступенчатая матрица отображается на экране с помощью функции print_row_reduced_matrix, что позволяет наглядно оценить изменения в структуре матрицы после применения метода Гаусса и определить лидирующие элементы.

Численное решение​

Численное решение системы уравнений представляет собой процесс определения приблизительных значений переменных, когда точное аналитическое решение недостижимо или нежелательно. Этот подход часто используется в ситуациях, когда необходимо получить численное приближение для решения системы.

from sympy import symbols, Eq, nsolve

# Определение переменных
x1, x2, x3, x4 = symbols('x1 x2 x3 x4')

# Определение системы уравнений
equations = [
Eq(x1 + x2 + x3 + x4, 0),
Eq(5*x1 - 3*x2 + 2*x3 - 8*x4, 1),
Eq(3*x1 + 5*x2 + x3 + 4*x4, 0),
Eq(4*x1 + 2*x2 + 3*x3 + x4, 3)
]

# Начальное предположение для численного решения
initial_guess = [0, 0, 0, 0]

# Нахождение численного решения
numerical_solution = nsolve(equations, (x1, x2, x3, x4), initial_guess)

# Вывод результатов
print("Численное решение:")
print(f"x1 = {numerical_solution[0]}")
print(f"x2 = {numerical_solution[1]}")
print(f"x3 = {numerical_solution[2]}")
print(f"x4 = {numerical_solution[3]}")
7c39541205b6a82eba067f68d67dcd4c.png

Код, который мы рассматриваем, задействует функцию nsolve из библиотеки SymPy для определения численного решения набора уравнений. Вначале определяются переменные x1, x2, x3 и x4 как символы. Далее, система уравнений представлена списком экземпляров Eq. Затем указывается начальное приближение для решения, представленное списком initial_guess.

Функция nsolve требует на вход список уравнений (equations), перечень символьных переменных (x1, x2, x3, x4) и начальное приближение (initial_guess). Она выполняет вычисление численного приближения значений переменных, которое максимально приближено к аналитическому решению, и возвращает полученные значения в форме списка.

Каждое найденное численное значение переменных x1, x2, x3 и x4 выводится отдельной строкой.

Метод наименьших квадратов​

Метод наименьших квадратов — это статистическая процедура, применяемая для оптимального приближения данных. Она заключается в поиске такой функции, которая минимизирует общую сумму квадратов разности между фактическими значениями и значениями, предсказанными моделью. Этот метод особенно полезен при работе с переопределёнными системами линейных уравнений, то есть когда число уравнений превосходит число неизвестных.

from sympy import symbols, Matrix

# Определение переменных
variables = symbols('x1 x2 x3 x4')

# Определение матрицы коэффициентов
coefficients_matrix = Matrix([
[1, 1, 1, 1],
[5, -3, 2, -8],
[3, 5, 1, 4],
[4, 2, 3, 1]
])

# Определение матрицы значений
constants_matrix = Matrix([0, 1, 0, 3])

# Решение системы с помощью метода наименьших квадратов
least_squares_solution = coefficients_matrix.solve_least_squares(constants_matrix)

# Вывод решения
print("Решение методом наименьших квадратов:", least_squares_solution)
4cb18cc8d85ee6a0144b7421e8d717e3.png

Здесь мы применяем функцию solve_least_squares от объекта Matrix библиотеки SymPy для решения системы линейных уравнений с избытком условий с использованием метода наименьших квадратов. Сначала вводятся переменные x1, x2, x3 и x4. Затем создаются матрицы: одна для коэффициентов (coefficients_matrix) и одна для свободных членов (constants_matrix) системы уравнений.

Функция solve_least_squares принимает эти две матрицы и находит вектор решений, который минимизирует сумму квадратов расхождений между произведением матрицы коэффициентов на вектор решений и реальными данными, содержащимися в матрице свободных членов.

Символьное решение​

Символьное решение системы уравнений — это подход, при котором переменные обрабатываются как символы, и применяются математические алгоритмы для получения их точных аналитических значений. В отличие от численных методов, при которых значения переменных находят в числовом виде, символьное решение выдаёт формулы для переменных, возможно, включающие в себя другие символы.

from sympy import symbols, Eq, solve

# Определение переменных
x1, x2, x3, x4 = symbols('x1 x2 x3 x4')

# Определение системы уравнений
equations = [
Eq(x1 + x2 + x3 + x4, 0),
Eq(5*x1 - 3*x2 + 2*x3 - 8*x4, 1),
Eq(3*x1 + 5*x2 + x3 + 4*x4, 0),
Eq(4*x1 + 2*x2 + 3*x3 + x4, 3)
]

# Решение системы символьно
symbolic_solution = solve(equations, (x1, x2, x3, x4))

# Вывод решения
print("Символьное решение:", symbolic_solution)

f17316ddd7f5d1c073e315a3b3748d97.png

В демонстрируемом коде начинают с определения символьных переменных x1, x2, x3 и x4, после чего уравнения системы формируются в коллекцию объектов Eq. Для символьного решения системы используется функция solve из библиотеки SymPy. Эта функция получает на вход список уравнений (equations) и список соответствующих символьных переменных (x1, x2, x3, x4), а возвращает словарь с аналитическими решениями для каждой из переменных.


В данной статье мы рассмотрели одну из функций библиотеки SymPy: решение систем линейных уравнений с использованием разных методов, выбор которых определяется конкретными задачами.

С точки зрения студента, важно не просто знание этих методов, важно глубокое понимание и умение применять их на практике, поскольку это может стать ценным опытом во многих областях науки и инженерии.

Выбор библиотеки SymPy для обзора был не случаен: она позволяет проводить символьные вычисления, обрабатывая выражения как серии символов, что значительно повышает точность математических операций. К тому же библиотека охватывает разнообразные сектора математики, включая решение уравнений и работу с матрицами. Дополнительным плюсом является её совместимость с другими инструментами, например с библиотекой Matplotlib, благодаря чему SymPy легко интегрируется для расширения своих возможностей.

 
Сверху