我正在做一个小科学项目,用ODE(常微分方程)模拟机械系统。
我想使用Euler方法来执行此操作,但是我认为我正在做一些工作,因为我创建的输出数据图不是应该的样子,并且我100%确信方程本身是正确的。您能看一下我的代码并告诉我我做错了什么吗?此代码会将CSV值输出到控制台和output.csv文件中。
测试:
class Calculus:
def __init__(self):
print "object created"
@staticmethod
def euler(f,y0,a,b,h):
"""y0 - temperatura poczatkowa, a-chwila poczatkowa czasu(warunek brzegowy) , b-chwila koncowa, h - krok"""
t,y = a,y0
time = [t]
value = [y]
while t <= b:
print "%6.3f,%6.3f" % (t,y)
t += h
y += h * f(t,y, h, value)
time.append(t)
value.append(y)
data = {'time' : time, 'value' : value}
return data
class Mechanical_system:
#Constructor
def __init__(self, momentum1, momentum2, b1, b2, N1, N2, w1, Tau, k):
self.system_parameters = {'momentum1': momentum1, 'momentum2': momentum2,
'b1': b1, 'b2' : b2,
'N1': N1, 'N2' : N2,
'w1' : w1,
'Tau' : Tau,
'k' : k};
def ODE(self, time, w, h, value):
"""ODE - ordinary differential equation describing our system"""
#first calculation will only have one value in the list, so we can't calculate delt_value = current - last
#thus, we need to assume that if there is no penault value (index error) let penault = 0
try:
penault = value[len(value) -2]
except IndexError:
penault = 0
momentum1 = self.system_parameters['momentum1']
momentum2 = self.system_parameters['momentum2']
b1 = self.system_parameters['b1']
b2 = self.system_parameters['b2']
N1 = self.system_parameters['N1']
N2 = self.system_parameters['N2']
Tau = self.system_parameters['Tau']
k = self.system_parameters['k']
dOmega = w - penault
dt = h
Omega = float(dOmega) / float(dt)
return (1.0 / (momentum1 + ((N1/N2)**2)*momentum2))*(Tau - (Omega)*(b1 + ((N1/N2)**2)*b2) - w * ((N1/N2)**2)*k)
if __name__ == "__main__":
"""reads values from input boxes and then calculates the value and plot value(time)"""
momentum1 = 1
momentum2 = 2
b1 = 0.5
b2 = 0.6
N1 = 10
N2 = 20
a = 0
b = 100
h = 0.1
w1 = 0
Tau = 100
k = 0.5
system1 = Mechanical_system(momentum1, momentum2, b1, b2, N1, N2, w1, Tau, k)
data = Calculus.euler(system1.ODE, w1, a, b, h)
##writing output to CSV file
f = open('output.csv', 'w')
for index in range(len(data['time'])):
f.write("%s,%s\n" % (data['time'][index], data['value'][index]))
f.close()
del system1
del data
最佳答案
在机械系统中使用Euler通常是个坏主意。探索该陈述最简单的测试案例是简单的振荡器x''+x=0
,您会发现系统的能量迅速增长。
对于一般的机械系统,您有一个运动方程m*x'' = F(t,x,x')
。这为您提供了矢量值系统
def f(t,y):
x,v = y
return [ v; 1/m*F(t,x,v) ]
请注意,系统功能中没有
h
或dt
。这对于ODE及其数值处理至关重要。轨迹遵循的方向场不取决于数值解法的细节。因此,请在相空间中将您的方程式重构为二阶方程式或一阶系统。
我认为您的方程式旨在
F(x,v) = Tau - (b1 + ((N1/N2)**2)*b2) * v - ((N1/N2)**2)*k * x
有质量
m = momentum1 + ((N1/N2)**2)*momentum2
或其某些缩放版本。所以某种弹簧在重力和摩擦力作用下。如果是这样,则在数值实现中将物理系统的加速度视为速度。这只能给出毫无意义的结果。
剥离这些类,可以像这样反映系统的第二顺序:
def euler(f,y0,v0,a,b,h):
t,y,v = a,y0,v0
time = [t]
value = [y]
while t <= b:
print "%6.3f,%6.3f" % (t,y)
accel = f(t,y,v)
t += h
y += h * v
v += h*accel
time.append(t)
value.append(y)
data = {'time' : time, 'value' : value}
return data
def ODE(t,y,v,param):
momentum1, momentum2, b1, b2, N1, N2, Tau, k = param
mass = momentum1 + ((N1/N2)**2)*momentum2
force = Tau - v*(b1 + ((N1/N2)**2)*b2) - y * ((N1/N2)**2)*k
return force/mass
momentum1 = 1
momentum2 = 2
b1 = 0.5
b2 = 0.6
N1 = 10.
N2 = 20.
Tau = 100
k = 0.5
a = 0
b = 100
h = 0.1
y0 = 0
v0 = 0
params = [momentum1, momentum2, b1, b2, N1, N2, Tau, k]
data = euler(lambda t,y,v: ODE(t,y,v,params),y0,v0,a,b,h)
关于python - 在python中实现Euler方法以解决ODE,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/30137545/