繁体   English   中英

在 sympy 中替换一个矩阵

[英]Substitute a matrix in sympy

我正在使用 pythTB 为石墨烯开发紧密结合的 model。 我想在计算中加入自旋元素。 rashba 跳跃项的 hamiltonian 使 pauli 自旋矩阵向量与站点跳跃向量交叉。

最初我创建了一个矩阵列表并将其与向量交叉,不幸的是这没有产生正确的结果(我认为在取向量叉积之后,然后取矩阵的叉积)。

接下来,我声明了 3 个符号“s_x”、“s_y”和“s_z”,并在我的 pauli 自旋矩阵向量中使用了这些符号而不是矩阵。 服用叉积后,我收到了正确的结果。 我遇到的问题是我无法将矩阵替换为我添加的变量符号。是否可以这样做? 还是我需要手动取叉积?

这是我的一些代码:

from __future__ import print_function
from pythtb import * # import TB model class
from sympy import symbols
import numpy as np
import matplotlib.pyplot as plt

# create list of pauli spin matrices 
sx = [[0., 1.],[1., 0.]]
sy = [[0., -1.j],[1.j, 0.]]
sz = [[1., 0.],[0., -1.]]
Id = [[1., 0.], [0., 1.]]
s_pauli = np.zeros((4, 2, 2), dtype=complex)
s_pauli = [Id, sx, sy, sz]

# create s_pauli without identity matrix
s_pau = np.zeros((3, 2, 2), dtype=complex)
s_pau = [ s_x, s_y, s_z]

ab00 = [ 0.5, 0.28867513, 0.]

sig_x_ab00 = np.cross( s_pau, ab00)

如果我打印sig_x_ab00[2] (这是我目前唯一感兴趣的),那么我会得到:

0.288675134594813*s_x - 0.5*s_y

获得该信息后,我想通过执行以下命令将s_pauli[1]替换为s_xs_pauli[2]s_y

sig_x_ab00_ = sig_x_ab00.subs(s_x, s_pauli[1])

我收到以下错误 output:

AttributeError: 'numpy.ndarray' object has no attribute 'subs'

我所做的一切有效吗? 或者有没有更好的方法来 go 关于这个?

非常感谢任何输入! 谢谢!

让我们运行您的代码,但查看每一步。 不要做假设。

我正在使用isympy交互环境; 带有ipython增强功能的sympy 我还进口np

In [4]: ab00 = [ 0.5, 0.28867513, 0.]                                           

In [5]: s_pauli                                                                 
Out[5]: 
[[[1.0, 0.0], [0.0, 1.0]],
 [[0.0, 1.0], [1.0, 0.0]],
 [[0.0, (-0-1j)], [1j, 0.0]],
 [[1.0, 0.0], [0.0, -1.0]]]

这是一个列表。 前面的np.zeros(...)表达式什么也不做。 在 Python 中,我们没有设置变量的“类型”。

我们可以从这个列表中创建一个数组:

In [6]: np.array(s_pauli)                                                       

s_pauli[1]有效,因为它只是列表索引。

以及添加的符号:

In [11]: s_x, s_y, s_z = symbols('s_x s_y s_z')                                 

In [12]: s_x                                                                    
Out[12]: sₓ

In [13]: s_pau = [ s_x, s_y, s_z]                                               

同样, s_pau是一个列表,而不是一个数组。 当在cross中使用时,它会变成一个数组:

In [14]: np.array(s_pau)                                                        
Out[14]: array([s_x, s_y, s_z], dtype=object)

请注意,这是一个 object dtype 数组,它仍然非常像一个列表。 一些基本的数学工作,因为像乘法和加法这样的数学是为符号定义的。 但是像np.lognp.sin这样的超越函数在 arrays 上不起作用。

cross只使用乘法和加法,因此它适用于这些 object arrays:

In [15]: sig = np.cross( s_pau, ab00)                                           

In [16]: sig                                                                    
Out[16]: array([-0.28867513*s_z, 0.5*s_z, 0.28867513*s_x - 0.5*s_y], dtype=object)

sig是一个 numpy 阵列。 它不是一个 sympy 表达式,也没有subs方法。 再次,密切关注正在发生的事情是值得的。

数组的元素是 sympy 表达式:

In [17]: sig[2]                                                                 
Out[17]: 0.28867513⋅sₓ - 0.5⋅s_y

In [20]: s2 = sig[2]                                                            

具有标量值的 subs 有效:

In [22]: s2.subs(s_x, 1)                                                        
Out[22]: 0.28867513 - 0.5⋅s_y

但没有清单

In [23]: s2.subs(s_x, s_pauli[1])                                               
Out[23]: 0.28867513⋅sₓ - 0.5⋅s_y

但是,如果我从中制作 sympy 矩阵:

In [24]: s_pauli[1]                                                             
Out[24]: [[0.0, 1.0], [1.0, 0.0]]

In [25]: Matrix(s_pauli[1])                                                     
Out[25]: 
⎡0.0  1.0⎤
⎢        ⎥
⎣1.0  0.0⎦

In [26]: s2.subs(s_x, Out[25])                                                  
Out[26]: 
           ⎡    0       0.28867513⎤
-0.5⋅s_y + ⎢                      ⎥
           ⎣0.28867513      0     ⎦

替换确实有效。

一般来说混合sympynumpy是命中注定的; 一些工作,几乎更多的是偶然而不是设计。 其他人没有。 sympy.lambdify是制作与 numpy arrays 一起使用的 function 的最可靠方法。

在这种情况下,我怀疑您最好使用cross的 sympy 版本,并进行sympy.Matrix替换。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM