RuntimeWarning:除算で無効な値が検出されました


96

「春のボール」モデルのオイラー法を使用してプログラムを作成する必要があります

from pylab import*
from math import*
m=0.1
Lo=1
tt=30
k=200
t=20
g=9.81
dt=0.01
n=int((ceil(t/dt)))
km=k/m
r0=[-5,5*sqrt(3)]
v0=[-5,5*sqrt(3)]
a=zeros((n,2))
r=zeros((n,2))
v=zeros((n,2))
t=zeros((n,2))
r[1,:]=r0
v[1,:]=v0
for i in range(n-1):
    rr=dot(r[i,:],r[i,:])**0.5
    a=-g+km*cos(tt)*(rr-L0)*r[i,:]/rr
    v[i+1,:]=v[i,:]+a*dt
    r[i+1,:]=r[i,:]+v[i+1,:]*dt
    t[i+1]=t[i]+dt

    #print norm(r[i,:])

plot(r[:,0],r[:,1])
xlim(-100,100)
ylim(-100,100)
xlabel('x [m]')
ylabel('y [m]')

show()

私はこのエラーを受け取り続けます:

a=-g+km*cos(tt)*(rr-L0)*r[i,:]/rr
RuntimeWarning: invalid value encountered in divide

私はそれを理解することができません、コードの何が問題になっていますか?


そのコード行の小さい項目のそれぞれで何が起こっているかを出力します。それをデバッグする唯一の方法です。
cppLearner 2013

2
あなたは持っているnanためのrrそのエラーを投げています、。の問題rrr[i,:]、場合によってはに等しいことに起因していarray([ nan, nan])ます。@CppLearnerが述べたように、コードをデバッグ(または記述)する最良の方法は、実装する前に各小さな部分をテストすることです。
コスモシス2013

回答:


160

あなたのコードは「ゼロ除算」または「NaNで除算」しようとしていると思います。あなたがそれを知っていて、それがあなたを煩わせたくないならば、あなたは試すことができます:

import numpy as np
np.seterr(divide='ignore', invalid='ignore')

詳細については、以下を参照してください。


76
with NP.errstate(divide='ignore',invalid='ignore'):コードブロックの警告を抑制したい場合に使用すると便利です。
GWW 2016年

8
ゼロ除算またはNaNを無視したいのはなぜですか?
X乗

7
@xsquared除算後、自分で値を正しく処理し、コードをユーザーに配布している(または警告が表示されるのにうんざりしている)場合。with np.errstate(...)処理されたケースだけでこれを安全に行うことができます。
reve_etrange 2018年

2
@reve_etrange一般的にゼロによる除算を無視するよりもはるかに受け入れられると思います。
X乗

1
これをエラーの原因となる行のに設定し、行の後で'warn'コマンドによっての通常の状態にリセットすることをおnp.seterr(divide='warn', invalid='warn')
勧めします

15

Pythonのインデックス作成は(1ではなく)0から始まるため、割り当て "r [1 、:] = r0"はrの2番目(つまりインデックス1)の要素を定義し、最初の(インデックス0)要素をゼロのペアとして残します。forループのiの最初の値は0であるため、rrはrの最初のエントリとそれ自体(0)の内積の平方根を取得し、次の行のrrで除算するとエラーがスローされます。


10

ゼロによる除算を防ぐために、div0エラーが発生する出力 'out'を事前に初期化できます。たとえばnp.where、条件に関係なく行全体が評価されるため、出力をカットしません。

事前初期化の例:

a = np.arange(10).reshape(2,5)
a[1,3] = 0
print(a)    #[[0 1 2 3 4], [5 6 7 0 9]]
a[0]/a[1]   # errors at 3/0
out = np.ones( (5) )  #preinit
np.divide(a[0],a[1], out=out, where=a[1]!=0) #only divide nonzeros else 1

4

あなたはrr0.0であるかもしれないことで割っています。rrがゼロかどうかを確認し、分母で使用する以外の合理的なことを行います。

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.