-
Notifications
You must be signed in to change notification settings - Fork 227
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
incorrect quartic root solver #1055
Comments
Confirmed. @NAThompson the function is failing when we get to here:
The equation being solved is x^2 + 273.75592674401321x + 18735.576856868389 = 0 which does have a root at ~ -136 see : https://www.wolframalpha.com/input?i=solve+x%5E2+%2B+273.75592674401321x+%2B+18735.576856868389+%3D+0 The graph here shows we are very close to missing the axis and having no real root: https://www.wolframalpha.com/input?i=plot+y+%3D+x%5E2+%2B+273.75592674401321x+%2B+18735.576856868389 so this may be a case of numerical error (something we will always hit eventually). |
The discriminant of the quadratic is negative, yet it still has a solution??? |
That polynomial never reaches 0 on the Reals: https://www.wolframalpha.com/input?i=minimize+x%5E2%2B273.75592674401321x%2B18735.576856868389 When interpreted as exact rational numbers, the minimum value is positive: |
@jzmaddock : I believe your suspicion of numerical error is correct. That said, >>> import numpy
>>> p = numpy.polynomial.Polynomial([0.24965766552610175, 273.7522788326969, 75042.069484941996, -547.5045576653938, 1.0])
>>> print(p)
0.24965767 + 273.75227883·x + 75042.06948494·x² - 547.50455767·x³ + 1.0·x⁴
>>> p.roots()
array([-1.82420204e-03 +0.j , -1.82370928e-03 +0.j ,
2.73754103e+02-10.13695958j, 2.73754103e+02+10.13695958j]) Moreover, though these roots are not well separated, the condition number of the rootfinding problem is not intractably large: >>> dpdx = p.deriv()
>>> condition_number = 1/abs(dpdx(-1.82370928e-03))
>>> condition_number
27.042429151092385 So there's a chance that we can use some trickery somewhere in the algorithm to improve the precision of some intermediate and recover these roots. @wztzjhn : Thanks for the report; this appears to be a real problem, though at this point I'm not sure I can guarantee successful resolution. The "correct" way to do this is to use the companion matrix, but this cannot distinguish between real and complex roots, whereas the algorithm employed here always finds the real roots. Will investigate more. |
Oi, the following diff fixes it: Will submit a fix later today; thanks @wztzjhn , @jzmaddock , and @pulver for triaging and making this so easy for me. @jzmaddock , @pulver , @mborland : Quick favor: Would one of you cast the coefficients to quad and see what the roots are? (I don't have quadmath on ARM.) I'm still about a million ulps off the Mathematica solution (relative error of 10^-9) and if Mathematica is using double precision under the hood I'm not sure which of us is closer to the truth. |
Previously, we took a square root, and then squared it later in the computation. In Issue #1055, we observed that this caused enough loss of accuracy to prevent finding real roots. Use the original value, instead of squaring the square root.
Previously, we took a square root, and then squared it later in the computation. In Issue #1055, we observed that this caused enough loss of accuracy to prevent finding real roots. Use the original value, instead of squaring the square root.
Hi,
I am testing the quartic root solver recently. While in most cases it gives the right answer, I see cases it doesn't do the job right. The following is a simple c++ code to show the error:
The output on my Ubuntu 22.04, with Boost 1.83.0, is:
To see the error more clearly, a Mathematica screenshot is shown below:
The text was updated successfully, but these errors were encountered: