I'm trying to lambdify big analytic expression with sp.Max(x, 0) inside. I want to use numpy to vectorize my calculations, so x is going to be an array. I need element-wise maximum values of x and 0. Still, sympy changes sp.Max to np.amax by default. It finds maximum along the axis, that's not what I need. "modules" keyword in lambdify doesn't work as I expect. I've tried:
import numpy as np
import sympy as sp
arr = np.array([1, 2, 3])
expr = sp.sin(x) + sp.Max(x, 0)
f = sp.lambdify(x, expr, modules=[{'Max': np.maximum}, 'numpy']) # docs say, priority of modules matters
help(f)
It gives:
Help on function _lambdifygenerated:
_lambdifygenerated(x)
Created with lambdify. Signature:
func(x)
Expression:
sin(x) + Max(0, x)
Source code:
def _lambdifygenerated(x):
return (sin(x) + amax((0,x)))
Imported modules:
sp.Max changed to amax for some reason.
If 'numpy' is not included into 'modules' list, it simply skips all other functions. I've also tried to swap dict and 'numpy' in list, but it haven't helped. Please, clarify, what's wrong? Is it a bug in sympy?
When using
lambdify
to create numpy functions intended to work vectorized, there often are subtle problems, especially when variables (x
) and constants (0
) are mixed.In this case,
sp.max
supposes all of its possible many parameters being single values.np.amax
gets the maximum of one flattened array.np.maximum
gets the element-wise maximum of two arrays. The problem here is that the constant0
doesn't automatically get expanded to a numpy array.My workaround would be to replace
sp.max
with a custom function based onsp.Piecewise
. Note that you would need a separate function if there would be more than 2 arguments tosp.max
.