I wanted to do the Newton-Raphson method with scipy with a multivariable system. So, i followed this documentation. And here is an example code where i tried to solved my problem:
import numpy as np
from scipy.optimize import newton
def f(x):
F = [0, 0]
F[0] = 5 * x[0] + 7 * x[1] - 6
F[1] = 10 * x[0] - 3 * x[1] - 46
return F
def jacobian(x):
F = np.zeros((2, 2))
F[0, 0] = 5
F[0, 1] = 7
F[1, 0] = 10
F[1, 1] = -3
return F
q = [10, -10]
result = newton(f, q, fprime=jacobian, maxiter=10000, full_output=1)
print(result)
When i run this program without the 'fprime' parameter (in the newton function), i get the correct answer:
result(root=array([ 3.99999999, -2.00000004]), converged=array([ True, True]), zero_der=array([False, False]))
But this is the secant method and i need the Newton method for a more complex problem. So when i run the code with the parameter 'fprime' i get the following error:
IndexError: too many indices for array: array is 1-dimensional, but 2 were indexed
Seeing this error i tried several solutions, like changing the matrix F from the jacobian function to a normal list:
def jacobian(x):
F = [0,0,0,0]
F[0] = 5
F[1] = 7
F[2] = 10
F[3] = -3
return F
Or changing all the variable to numpy arrays, but i keep getting similar errors with index.
I want know if it is posible to do this system of equations with this scipy function and if not, i would like other alternatives with python.
The traceback is
Cell In[46], line 26
result = newton(f, q, fprime=jacobian, maxiter=10000, full_output=1)
File ~\miniconda3\lib\site-packages\scipy\optimize\_zeros_py.py:277 in newton
return _array_newton(func, x0, fprime, args, tol, maxiter, fprime2,
File ~\miniconda3\lib\site-packages\scipy\optimize\_zeros_py.py:401 in _array_newton
dp = fval[nz_der] / fder[nz_der]
IndexError: too many indices for array: array is 1-dimensional, but 2 were indexed
Changing the ipython
%xmode` to verbose, the error line, with context and variables is:
File ~\miniconda3\lib\site-packages\scipy\optimize\_zeros_py.py:401, in _array_newton(func=<function f>, x0=[10, -10], fprime=<function jacobian>, args=(), tol=1.48e-08, maxiter=10000, fprime2=None, full_output=1)
399 break
400 # Newton step
400 # Newton step
--> 401 dp = fval[nz_der] / fder[nz_der]
nz_der = array([[ True, True],
[ True, True]])
fval = array([-26, 84])
fder = array([[ 5., 7.],
[10., -3.]])
I think fval
is the 1d array produced by f(q)
. fder
is the 2d array produced by jacobian
, and nz_der
looks like a matching shape boolean.
You/we need to review the newton
docs to see what shape of a result the fprime
must be.
The docs aren't specific about what fprime
must return, but from that error, and the examples like fprime=lambda x: 3 * x**2)
, it must be a 1d array like f(q)
, the same shape as in guess q
. You are trying to provide the 2d dz/dx.
Using np.array([5,7,10,-3])
has indexing problem at the same place, only with a (4,) boolean instead of the (2,2).
From your linked docs:
If x0 is a sequence with more than one item, newton returns an array: the zeros of the function from each (scalar) starting point in x0. In this case, func must be vectorized to return a sequence or array of the same shape as its first argument. If fprime (fprime2) is given, then its return must also have the same shape: each element is the first (second) derivative of func with respect to its only variable evaluated at each element of its first argument.