python - 模拟电子运动——python中自适应步长的微分方程

标签 python scipy physics ode differential-equations

我正在尝试模拟强激光的振荡电场如何插入接近 +1 ionic 库仑势的电子。激光场是

E = Eo sin(wt),在 y 方向。

库仑势是

F = ke q1*q2/r^2,在 r 方向。

强电场导调用子tunnel ionize ,因此电子的初始条件是在 y 方向上从原子中位移。然后,电子被激光场来回插入,并有机会与库仑势相互作用。我想模拟库仑势如何影响电子的飞行。模拟需要在三个维度上进行,因为我最终想要包括更复杂的激光场,这些激光场将电子推向 x 和 y 方向,电子可以从 z 方向的动量开始。

起初,我认为这很容易。下面是我用来以非常小的步长(1e-18 秒)步进时间的代码。当电子不在 ionic 附近时,这很好用。然而,对于靠近 ionic 通过的电子,结果在很大程度上取决于模拟中使用的时间步长。如果我将时间步变小,计算将花费很长时间。

因此,我认为在这种情况下我应该使用自适应时间步长。此外,据我所读,Runge-Kutta 方法应该优于我正在使用的简单方法。但是,我认为 scipy.odeint 不适用于三维。关于如何提高这些模拟的准确性和速度的任何想法?

下图显示了时间步长如何对结果产生巨大影响(坏事): enter image description here

这是我的代码:

import numpy as np
import matplotlib.pyplot as plt

q = 1.602e-19       #Coulombs   Charge of electron
h_bar = 1.054e-34   #J*s        Plank's Constant div by 2Pi
c = 3.0e8           #m/s        Speed of light
eo = 8.8541e-12     #C^2/(Nm^2) Permittivity of vacuum
me = 9.109e-31      #kg         Mass of electron
ke = 8.985551e9     #N m^2 C-2  Coulomb's constant

def fly_trajectory(wavelength,intensity,tb=0,pulseFWHM=40.e-15,
                   final_time=100e-15,time_step=.001e-15,Ip=15.13,v0=(2e4,0,0)):
    #Intensity is in w/cm2. Ip is in eV. Otherwise it's SI units throughout.
    #The electric field of the later is in the y-direction

    Ip = 15.13 * q  #Calculate the ionization potential of the atom in Joules
    Eo = np.sqrt(2*intensity*10**4/(c*8.85e-12)) # Electric field in V/m
    w = c/wavelength * 2. * np.pi #Angular frequency of the laser

    times = np.arange(tb,final_time,time_step)
    Ey = Eo*np.sin(w*times) * np.exp(-times**2/(2*(pulseFWHM / 2.35482)**2)) 
    Eb = Ey[0] #E-field at time of birth (time of tunneling)

    if Eb == 0: return 0,0 #No field --> no electrons
    tunnel_position = -Ip / (Eb*q)    
    x,y,z = 0,tunnel_position,0
    vx,vy,vz = v0

    y_list = np.zeros(len(times)) #store the y-values for later

    for index in range(0,len(times)):
        r = np.sqrt(x**2+y**2+z**2)
        rx = x/r; ry = y/r; rz=z/r

        Fcy = -q**2 * ke/(r**2) * ry
        ay = Ey[index]*(-q)/me + Fcy/me #only y includes the laser
        vy = vy + ay*time_step 
        y = y + vy * time_step

        Fcx = -q**2 * ke/(r**2) * rx
        ax = (-q)/me + Fcx/me
        vx = vx + ax*time_step  
        x = x + vx * time_step

        Fcz = -q**2 * ke/(r**2) * rz
        az = (-q)/me + Fcz/me
        vz = vz + az*time_step 
        z = z + vz * time_step

        y_list[index] = y

    return times,y_list

for tb in np.linspace(0.25*2.66e-15,0.5*2.66e-15,5):
    print tb
    times,ys = fly_trajectory(800e-9,2e14,tb=tb,time_step=.01e-15)
    plt.plot(times,ys,color='r')

    times,ys = fly_trajectory(800e-9,2e14,tb=tb,time_step=.001e-15)
    plt.plot(times,ys,color='b')


#plot legend and labels:
plt.plot(0,0,color='r',label='10e-18 sec step')
plt.plot(0,0,color='b',label='1e-18 sec step')
plt.xlim(0,10e-15); plt.ylim(-1e-8,1e-8)
leg = plt.legend(); leg.draw_frame(False)
plt.xlabel('Time (sec)')
plt.ylabel('Y-distance (meters)')
plt.show()

最佳答案

正如 Warren Weckesser 所建议的,我可以简单地按照耦合质量- Spring 系统的 Scipy 食谱进行操作。首先,我需要将“右侧”方程式写成:

x'  = vx
y'  = vy
z'  = vz
vx' = Ac*x/r
vy' = Ac*y/r + q*E/m
vz' = Ac*z/r 

其中 Ac=keq^2/(mr^2) 是由于库仑势引起的加速度的大小,E 是激光的随时间变化的电场。然后,我可以使用 scipy.integrate.odeint找到解决方案。这比我之前使用的方法更快、更可靠。

这是使用 odeint 后的电子轨迹。现在它们都没有疯狂地飞走: enter image description here

代码如下:

import numpy as np
import matplotlib.pyplot as plt
import scipy.integrate

q  = 1.602e-19    #Coulombs   Charge of electron
c  = 3.0e8        #m/s        Speed of light
eo = 8.8541e-12   #C^2/(Nm^2) Permittivity of vacuum
me = 9.109e-31    #kg         Mass of electron
ke = 8.985551e9   #N m^2 C-2  Coulomb's constant

def tunnel_position(tb,intensity,wavelength,pulseFWHM,Ip):
    Ip = 15.13 * q  
    Eb = E_laser(tb,intensity,wavelength,pulseFWHM) 
    return -Ip / (Eb*q) 

def E_laser(t,intensity,wavelength,pulseFWHM):
    w = c/wavelength * 2. * np.pi #Angular frequency of the laser
    Eo = np.sqrt(2*intensity*10**4/(c*8.85e-12)) # Electric field in V/m
    return Eo*np.sin(w*t) * np.exp(-t**2/(2*(pulseFWHM / 2.35482)**2))

def vectorfield(variables,t,params):
    x,y,z,vx,vy,vz = variables
    intensity,wavelength,pulseFWHM,tb = params
    r = np.sqrt(x**2+y**2+z**2)
    Ac = -ke*q**2/(r**2*me)
    return [vx,vy,vz,
         Ac*x/r,
         Ac*y/r + q/me * E_laser((t-tb),intensity,wavelength,pulseFWHM),
         Ac*z/r]

Ip  = 15.13   # Ionization potential of Argon eV
intensity  = 2e14
wavelength = 800e-9
pulseFWHM  = 40e-15

period  = wavelength/c
t = np.linspace(0,20*period,10000)

birth_times = np.linspace(0.01*period,0.999*period,50)
max_field   = np.max(np.abs(E_laser(birth_times,intensity,wavelength,pulseFWHM)))

for tb in birth_times:
    x0  = 0 
    y0  = tunnel_position(tb,intensity,wavelength,pulseFWHM,Ip)
    z0  = 0
    vx0 = 2e4
    vy0 = 0
    vz0 = 0

    p = [intensity,wavelength,pulseFWHM,tb]
    w0 = [x0,y0,z0,vx0,vy0,vz0]

    solution,info = scipy.integrate.odeint(vectorfield,w0,t, args=(p,),full_output=True)
    print 'Tb: %.2f fs - smallest step : %.05f attosec'%((tb*1e15),np.min(info['hu'])*1e18)

    y = solution[:,1]

    importance = (np.abs(E_laser(tb,intensity,wavelength,pulseFWHM))/max_field)
    plt.plot(t,y,alpha=importance*0.8,lw=1)

plt.xlabel('Time (sec)')
plt.ylabel('Y-distance (meters)')

plt.show()

关于python - 模拟电子运动——python中自适应步长的微分方程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26063269/

相关文章:

javascript - 当惯性滚动算法接近顶部或底部时,我该如何减速?

python - python的空白问题

machine-learning - python中适合稀疏高维特征的理想分类器(具有层次分类)

python - 我可以在 D(而不是 C)中创建 Python 扩展模块吗

python - 共轭梯度模块中的矩阵函数

python - 在 SciPy 中多核上运行 SVM 代码?

physics - Box2d 中的摩擦

javascript - 了解 EaselJS 的动画/物理/数学实现

python - 在Python中使用重复的函数调用来循环某些东西(即列表)是否合适?

python - Sqlite3不从数据库读取