Các vấn đề khi thay thế một ma trận trong một đa thức
Ví dụ: let
M = Matrix([[1,2],[3,4]]) # and
p = Poly(x**3 + x + 1) # then
p.subs(x,M).expand()
đưa ra lỗi:
TypeError: không thể thêm <class'sympy.matrices.immutable.ImmutableDenseMatrix '> và <class' evaly.core.numbers.One '>
điều này rất hợp lý vì hai số hạng đầu tiên trở thành ma trận nhưng số hạng cuối cùng (số hạng hằng) không phải là ma trận mà là một đại lượng vô hướng. Để khắc phục tình huống này, tôi đã thay đổi đa thức thành
p = Poly(x**3 + x + x**0) # then
lỗi tương tự vẫn tồn tại. Tôi có bắt buộc phải nhập biểu thức bằng tay, thay x bằng M không? Trong ví dụ này, đa thức chỉ có ba số hạng nhưng trong thực tế tôi gặp (đa thức nhiều biến với) hàng chục số hạng.
Trả lời
Vì vậy, tôi nghĩ câu hỏi chủ yếu xoay quanh khái niệm đa thức Ma trận :
(trong đó P là một đa thức và A là một ma trận)
Tôi nghĩ rằng điều này đang nói rằng số hạng tự do là một số, và nó không thể được thêm vào với phần còn lại là một ma trận, về hiệu quả, phép toán cộng là không xác định giữa hai loại đó.
TypeError: không thể thêm <class'sympy.matrices.immutable.ImmutableDenseMatrix '> và <class' evaly.core.numbers.One '>
Tuy nhiên, điều này có thể được phá vỡ bằng cách xác định một hàm đánh giá đa thức ma trận cho một ma trận cụ thể. Sự khác biệt ở đây là chúng ta đang sử dụng ma trận lũy thừa , vì vậy chúng tôi một cách chính xác tính toán hạn tự do của đa thức ma trận a_0 * I
mà I=A^0
là ma trận sắc của hình yêu cầu:
from sympy import *
x = symbols('x')
M = Matrix([[1,2],[3,4]])
p = Poly(x**3 + x + 1)
def eval_poly_matrix(P,A):
res = zeros(*A.shape)
for t in enumerate(P.all_coeffs()[::-1]):
i, a_i = t
res += a_i * (A**i)
return res
eval_poly_matrix(p,M)
Đầu ra:
Trong ví dụ này, đa thức chỉ có ba số hạng nhưng trong thực tế tôi gặp (đa thức nhiều biến với) hàng chục số hạng.
Hàm eval_poly_matrix
trên có thể được mở rộng để hoạt động đối với các đa thức nhiều biến bằng cách sử dụng .monoms()phương pháp trích xuất các đơn thức có hệ số khác không , như sau:
from sympy import *
x,y = symbols('x y')
M = Matrix([[1,2],[3,4]])
p = poly( x**3 * y + x * y**2 + y )
def eval_poly_matrix(P,*M):
res = zeros(*M[0].shape)
for m in P.monoms():
term = eye(*M[0].shape)
for j in enumerate(m):
i,e = j
term *= M[i]**e
res += term
return res
eval_poly_matrix(p,M,eye(M.rows))
Lưu ý: Có thể thực hiện một số kiểm tra độ tỉnh táo, xử lý các trường hợp cạnh và tối ưu hóa:
- Số lượng biến có trong đa thức liên quan đến số lượng ma trận được truyền dưới dạng tham số (ma trận trước không bao giờ được lớn hơn ma trận sau và nếu nó thấp hơn một số logic cần có để xử lý điều đó, tôi chỉ xử lý trường hợp này khi cả hai bằng nhau)
- Tất cả các ma trận cần phải là hình vuông theo định nghĩa của đa thức ma trận
- Một cuộc thảo luận về một phiên bản đa biến của các đặc điểm của quy tắc Horner trong phần nhận xét của câu hỏi này . Điều này có thể hữu ích để giảm thiểu số lượng phép nhân ma trận.
- Xử lý thực tế rằng trong một đa thức Ma trận
x*y
khác vớiy*x
vì phép nhân ma trận là không giao hoán . Rõ ràng các hàm poly trong giao hưởng không hỗ trợ các biến không giao hoán , nhưng bạn có thể xác định các ký hiệu vớicommutative=False
và dường như có một con đường chuyển tiếp ở đó
Về điểm thứ 4 ở trên, có hỗ trợ cho các biểu thức Ma trận trong SymPy và điều đó có thể giúp ở đây:
from sympy import *
from sympy.matrices import MatrixSymbol
A = Matrix([[1,2],[3,4]])
B = Matrix([[2,3],[3,4]])
X = MatrixSymbol('X',2,2)
Y = MatrixSymbol('Y',2,2)
I = eye(X.rows)
p = X**2 * Y + Y * X ** 2 + X ** 3 - I
display(p)
p = p.subs({X: A, Y: B}).doit()
display(p)
Đầu ra:
Để biết thêm các phát triển về tính năng này, hãy theo dõi # 18555