i am trying to get root of a function using Newton's algorithm on python. I have a runtime error even if i change the precision level. can you kindly help me understand how can i improve it?
Best, GB
Below my 'simple' code and the root finding part:
from scipy.stats import norm
import numpy as np
import matplotlib.pyplot as plt
import scipy.optimize as opt
def vega_callspread(r,S,T,d,sigma,q,K1,K2):
d1 = (np.log(S/K1)+(r+sigma*sigma/2)*T)/(sigma*np.sqrt(T))
d2 = (np.log(S/K2)+(r+sigma*sigma/2)*T)/(sigma*np.sqrt(T))
u1 = S*np.exp(-q*T)*norm.pdf(d1,0,1)
u2 = S*np.exp(-q*T)*norm.pdf(d2,0,1)
return u1-u2;
x0=112
r=0
T=1
d=0
sigma=0.2
q=0
K1=110
K2=130
res2= opt.newton(vega_callspread, x0, args=(r,T,d,sigma,q,K1,K2,),tol=10**(-1),maxiter=1000000)
the error i get is: res2= opt.zeros.newton(vega_callspread, x0, args=(r,T,d,sigma,q,K1,K2,),tol=10**(-1),maxiter=1000000)
/Users/anaconda/lib/python3.5/site-packages/scipy/optimize/zeros.py:173: RuntimeWarning: Tolerance of 0.011300000000005639 reached
warnings.warn(msg, RuntimeWarning)
It's hard to give advice with such sparse context. But some remarks:
A:
You are not using newton, as described here:
B:
Your error comes from here i would say:
As those tol-values are hardcoded, i suppose this should not happen!
Meaning: there is probably something wrong with your code!
C:
Let's try the slower but more safe bisection-method:
Output:
That's a bad sign.
D (follows C):
Your func reads:
and you call with:
There is no S!
So either you are dangerously renaming variables or there is some error on your end!
The value of
S
is the problem inlog(S/K1)
and co.It's not about
S/K1
but about this:which i got from this SO-answer.
You can try to interpret what this is doing to your function now as this log-eval will get the value of
-np.inf
.E How is scipy handling this (follows D):
I'm too lazy to read the docs/sources of args-handling, but let's check something (add a print to the func; use bisection as earlier):
Output:
So it seems arg-handling is by arg-name (and not just param-position in the call; i'm missing the correct programming-language terms here; again lazy!)
This (S=0) is probably very bad (and not what you want) and this is an error on your end!
EDIT
After your comment, it seems you try to optimize
S
. That made it clear to me, that you are using some optimization-algorithm optimizingx
where you don't havex
in your function!I'm not analyzing your task here, but you probably want to make your function to use some
x
(initialized byx0
) as this is the general idea ofscipy.optimize
. We can keep the nameS
, but it needs to be the first parameter of the function. This is all explained in the docs.So:
Output: