Coverage for pygeodesy/fmath.py : 96%
 
         
         
    Hot-keys on this page
r m x p toggle line displays
j k next/prev highlighted chunk
0 (zero) top of page
1 (one) first highlighted chunk
| 
 # -*- coding: utf-8 -*- 
 ''' # make sure int/int division yields float quotient, see .basics 
 isscalar, len2, remainder as _remainder _xError _few_, _h_, _negative_, _not_scalar_, _singular_, \ _too_, _0_0, _0_5, _1_0, _1_3rd, _2_3rd, _N_1_0, _1_5 # from pygeodesy.streprs import Fmt, unstr # from .fsums 
 
 
 # sqrt(2) <https://WikiPedia.org/wiki/Square_root_of_2> 
 
 '''Precision dot product. ''' '''New L{Fdot} precision dot product M{sum(a[i] * b[i] for i=0..len(a))}. 
 @arg a: Iterable, list, tuple, etc. (C{scalar}s). @arg b: Other values (C{scalar}s), all positional. 
 @raise OverflowError: Partial C{2sum} overflow. 
 @raise LenError: Unequal C{len(B{a})} and C{len(B{b})}. 
 @see: Function L{fdot} and method L{Fsum.fadd}. ''' 
 
 '''Precision polynomial evaluation using the Horner form. ''' '''New L{Fhorner} evaluation of the polynomial M{sum(cs[i] * x**i for i=0..len(cs))}. 
 @arg x: Polynomial argument (C{scalar}). @arg cs: Polynomial coeffients (C{scalar}s), all positional. 
 @raise OverflowError: Partial C{2sum} overflow. 
 @raise TypeError: Non-scalar B{C{x}}. 
 @raise ValueError: Non-finite B{C{x}}. 
 @see: Function L{fhorner} and methods L{Fsum.fadd} and L{Fsum.fmul}. ''' # assert self._ps is ps 
 
 '''Precision polynomial evaluation. ''' '''New L{Fpolynomial} evaluation of the polynomial M{sum(cs[i] * x**i for i=0..len(cs))}. 
 @arg x: Polynomial argument (C{scalar}). @arg cs: Polynomial coeffients (C{scalar}s), all positional. 
 @raise OverflowError: Partial C{2sum} overflow. 
 @raise TypeError: Non-scalar B{C{x}}. 
 @raise ValueError: Non-finite B{C{x}}. 
 @see: Function L{fpolynomial} and method L{Fsum.fadd}. ''' 
 
 
 def cbrt2(x3): '''Compute the cube root I{squared} M{x3**(2/3)}. ''' return cbrt(x3)**2 # cbrt(-0.0*2) == -0.0 
 except ImportError: # Python 3.10- 
 def cbrt(x3): '''Compute the cube root M{x3**(1/3)}. 
 @arg x3: Value (C{scalar}). 
 @return: Cubic root (C{float}). 
 @see: Functions L{cbrt2} and L{sqrt3}. ''' # <https://archive.lib.MSU.edu/crcmath/math/math/r/r021.htm> # simpler and more accurate than Ken Turkowski's CubeRoot, see # <https://People.FreeBSD.org/~lstewart/references/apple_tr_kt32_cuberoot.pdf> return _copysign(pow(fabs(x3), _1_3rd), x3) # cbrt(-0.0) == -0.0 
 def cbrt2(x3): # PYCHOK attr '''Compute the cube root I{squared} M{x3**(2/3)}. 
 @arg x3: Value (C{scalar}). 
 @return: Cube root I{squared} (C{float}). 
 @see: Functions L{cbrt} and L{sqrt3}. ''' return pow(fabs(x3), _2_3rd) # XXX pow(fabs(x3), _1_3rd)**2 
 
 '''I{Appoximate} the norm M{sqrt(x**2 + y**2)} by M{max(abs(x), abs(y)) + min(abs(x), abs(y)) * 0.4142...}. 
 @arg x: X component (C{scalar}). @arg y: Y component (C{scalar}). 
 @return: Appoximate norm (C{float}). 
 @see: Function L{euclid_}. ''' 
 
 '''I{Appoximate} the norm M{sqrt(sum(x**2 for x in xs))} by cascaded L{euclid}. 
 @arg xs: X arguments, positional (C{scalar}s). 
 @return: Appoximate norm (C{float}). 
 @see: Function L{euclid}. ''' # e = euclid(x, e) 
 
 '''Fast approximation of L{pygeodesy.acos1}C{(B{x})}. 
 @see: U{ShaderFastLibs.h<https://GitHub.com/michaldrobot/ ShaderFastLibs/blob/master/ShaderFastMathLib.h>}. ''' r = PI - r else: r = PI if x < 0 else _0_0 
 
 def fasin1(x): # PYCHOK no cover '''Fast approximation of L{pygeodesy.asin1}C{(B{x})}. 
 @see: L{facos1}. ''' return PI_2 - facos1(x) 
 
 '''Fast approximation of C{atan(B{x})}. ''' else: r = -r 
 
 '''Fast approximation of C{atan(B{x})} for C{0 <= B{x} <= 1}, I{unchecked}. 
 @see: U{ShaderFastLibs.h<https://GitHub.com/michaldrobot/ShaderFastLibs/ blob/master/ShaderFastMathLib.h>} and U{Efficient approximations for the arctangent function<http://www-Labs.IRO.UMontreal.CA/ ~mignotte/IFT2425/Documents/EfficientApproximationArctgFunction.pdf>}, IEEE Signal Processing Magazine, 111, May 2006. ''' # Eq (9): PI_4 * x - x * (abs(x) - 1) * (0.2447 + 0.0663 * abs(x)), for -1 < x < 1 # PI_4 * x - (x**2 - x) * (0.2447 + 0.0663 * x), for 0 < x - 1 # x * (1.0300981633974482 + x * (-0.1784 - x * 0.0663)) 
 
 '''Fast approximation of C{atan2(B{y}, B{x})}. 
 @see: U{fastApproximateAtan(x, y)<https://GitHub.com/CesiumGS/cesium/blob/ master/Source/Shaders/Builtin/Functions/fastApproximateAtan.glsl>} and L{fatan1}. ''' else: # a == b == 0 return _0_0 r = PI - r r = -r 
 
 '''Return the average of two values. 
 @arg v1: One value (C{scalar}). @arg v2: Other value (C{scalar}). @kwarg f: Optional fraction (C{float}). 
 @return: M{v1 + f * (v2 - v1)} (C{float}). ''' # @raise ValueError: Fraction out of range. # ''' # if not 0 <= f <= 1: # XXX restrict fraction? # raise _ValueError(fraction=f) # v1 + f * (v2 - v1) == v1 * (1 - f) + v2 * f 
 
 '''Return the precision dot product M{sum(a[i] * b[i] for i=0..len(a))}. 
 @arg a: Iterable, list, tuple, etc. (C{scalar}s). @arg b: All positional arguments (C{scalar}s). 
 @return: Dot product (C{float}). 
 @raise LenError: Unequal C{len(B{a})} and C{len(B{b})}. 
 @see: Class L{Fdot} and U{Algorithm 5.10 B{DotK} <https://www.TUHH.De/ti3/paper/rump/OgRuOi05.pdf>}. ''' 
 
 '''Return the precision dot product M{start + sum(a[i] * b[i] * c[i] for i=0..len(a))}. 
 @arg a: Iterable, list, tuple, etc. (C{scalar}s). @arg b: Iterable, list, tuple, etc. (C{scalar}s). @arg c: Iterable, list, tuple, etc. (C{scalar}s). @kwarg start: Optional bias (C{scalar}). 
 @return: Dot product (C{float}). 
 @raise LenError: Unequal C{len(B{a})}, C{len(B{b})} and/or C{len(B{c})}. 
 @raise OverflowError: Partial C{2sum} overflow. ''' 
 
 raise LenError(fdot3, a=len(a), b=len(b), c=len(c)) 
 
 
 '''Evaluate the polynomial M{sum(cs[i] * x**i for i=0..len(cs))} using the Horner form. 
 @arg x: Polynomial argument (C{scalar}). @arg cs: Polynomial coeffients (C{scalar}s). 
 @return: Horner value (C{float}). 
 @raise OverflowError: Partial C{2sum} overflow. 
 @raise TypeError: Non-scalar B{C{x}}. 
 @raise ValueError: No B{C{cs}} coefficients or B{C{x}} is not finite. 
 @see: Function L{fpolynomial} and class L{Fhorner}. ''' 
 
 '''Interpolate using using U{Inverse Distance Weighting <https://WikiPedia.org/wiki/Inverse_distance_weighting>} (IDW). 
 @arg xs: Known values (C{scalar}s). @arg ds: Non-negative distances (C{scalar}s). @kwarg beta: Inverse distance power (C{int}, 0, 1, 2, or 3). 
 @return: Interpolated value C{x} (C{float}). 
 @raise LenError: Unequal or zero C{len(B{ds})} and C{len(B{xs})}. 
 @raise ValueError: Invalid B{C{beta}}, negative B{C{ds}} value, weighted B{C{ds}} below L{EPS}. 
 @note: Using C{B{beta}=0} returns the mean of B{C{xs}}. ''' raise LenError(fidw, xs=n, ds=d) 
 if isnear0(d): # PYCHOK no cover n = Fmt.PAREN(fsum='ds') raise _ValueError(n, d, txt=_singular_) else: # b == 0 x = fsum(xs, floats=True) / n # fmean(xs) elif d < 0: # PYCHOK no cover n = Fmt.SQUARE(ds=ds.index(d)) raise _ValueError(n, d, txt=_negative_) 
 
 '''Compute the accurate mean M{sum(xs[i] for i=0..len(xs)) / len(xs)}. 
 @arg xs: Values (C{scalar} or L{Fsum} instances). 
 @return: Mean value (C{float}). 
 @raise OverflowError: Partial C{2sum} overflow. 
 @raise ValueError: No B{C{xs}} values. ''' raise _ValueError(xs=xs) 
 
 '''Compute the accurate mean M{sum(xs[i] for i=0..len(xs)) / len(xs)}. 
 @arg xs: Values (C{scalar} or L{Fsum} instances). 
 @return: Mean value (C{float}). 
 @raise OverflowError: Partial C{2sum} overflow. 
 @raise ValueError: No B{C{xs}} values. ''' 
 
 '''Evaluate the polynomial M{sum(cs[i] * x**i for i=0..len(cs))}. 
 @arg x: Polynomial argument (C{scalar}). @arg cs: Polynomial coeffients (C{scalar}s), all positional. 
 @return: Polynomial value (C{float}). 
 @raise OverflowError: Partial C{2sum} overflow. 
 @raise TypeError: Non-scalar B{C{x}}. 
 @raise ValueError: No B{C{cs}} coefficients or B{C{x}} is not finite. 
 @see: Function L{fhorner} and class L{Fpolynomial}. ''' 
 
 '''Return a series of powers M{[x**i for i=1..n]}. 
 @arg x: Value (C{scalar}). @arg n: Highest exponent (C{int}). @kwarg alts: Only alternating powers, starting with this exponent (C{int}). 
 @return: Powers of B{C{x}} (C{float}s or C{int}s). 
 @raise TypeError: Non-scalar B{C{x}} or B{C{n}} not C{int}. 
 @raise ValueError: Non-finite B{C{x}} or non-positive B{C{n}}. ''' raise _IsnotError(int.__name__, n=n) raise _ValueError(n=n) 
 
 # ps[alts-1::2] chokes PyChecker 
 
 
 except ImportError: 
 def fprod(xs, start=1): '''Iterable product, like C{math.prod} or C{numpy.prod}. 
 @arg xs: Terms to be multiplied, an iterable, list, tuple, etc. (C{scalar}s). @kwarg start: Initial term, also the value returned for an empty B{C{xs}} (C{scalar}). 
 @return: The product (C{float}). 
 @see: U{NumPy.prod<https://docs.SciPy.org/doc/ numpy/reference/generated/numpy.prod.html>}. ''' return freduce(_mul, xs, start) 
 
 '''Generate a range of C{float}s. 
 @arg start: First value (C{float}). @arg number: The number of C{float}s to generate (C{int}). @kwarg step: Increment value (C{float}). 
 @return: A generator (C{float}s). 
 @see: U{NumPy.prod<https://docs.SciPy.org/doc/ numpy/reference/generated/numpy.arange.html>}. ''' raise _IsnotError(int.__name__, number=number) 
 
 except ImportError: try: freduce = reduce # PYCHOK expected except NameError: # Python 3+ 
 def freduce(f, xs, *start): '''For missing C{functools.reduce}. ''' if start: r = v = start[0] else: r, v = 0, MISSING for v in xs: r = f(r, v) if v is MISSING: raise _TypeError(xs=(), start=MISSING) return r 
 
 '''Remainder in range C{[-B{y / 2}, B{y / 2}]}. 
 @arg x: Numerator (C{scalar}). @arg y: Modulus, denominator (C{scalar}). 
 @return: Remainder (C{scalar}, preserving signed 0.0) or C{NAN} for any non-finite B{C{x}}. 
 @raise ValueError: Infinite or near-zero B{C{y}}. 
 @see: I{Karney}'s U{Math.remainder<https://PyPI.org/ project/geographiclib/>} and Python 3.7+ U{math.remainder<https://docs.Python.org/3/ library/math.html#math.remainder>}. ''' # with Python 2.7.16 and 3.7.3 on macOS 10.13.6 and # with Python 3.10.2 on macOS 12.2.1 M1 arm64 native # fmod( 0, 360) == 0.0 # fmod( 360, 360) == 0.0 # fmod(-0, 360) == 0.0 # fmod(-0.0, 360) == -0.0 # fmod(-360, 360) == -0.0 # however, using the % operator ... # 0 % 360 == 0 # 360 % 360 == 0 # 360.0 % 360 == 0.0 # -0 % 360 == 0 # -360 % 360 == 0 == (-360) % 360 # -0.0 % 360 == 0.0 == (-0.0) % 360 # -360.0 % 360 == 0.0 == (-360.0) % 360 
 # On Windows 32-bit with python 2.7, math.fmod(-0.0, 360) # == +0.0. This fixes this bug. See also Math::AngNormalize # in the C++ library, Math.sincosd has a similar fix. except Exception as e: t = fremainder.__name__ raise _xError(e, unstr(t, x, y)) else: # handle x INF and NINF as NAN r = NAN 
 
 if _sys_version_info2 < (3, 8): # PYCHOK no cover from math import hypot # OK in Python 3.7- 
 def hypot_(*xs): '''Compute the norm M{sqrt(sum(x**2 for x in xs))}. 
 Similar to Python 3.8+ n-dimension U{math.hypot <https://docs.Python.org/3.8/library/math.html#math.hypot>}, but exceptions, C{nan} and C{infinite} values are handled differently. 
 @arg xs: X arguments (C{scalar}s), all positional. 
 @return: Norm (C{float}). 
 @raise OverflowError: Partial C{2sum} overflow. 
 @raise ValueError: Invalid or no B{C{xs}} values. 
 @note: The Python 3.8+ Euclidian distance U{math.dist <https://docs.Python.org/3.8/library/math.html#math.dist>} between 2 I{n}-dimensional points I{p1} and I{p2} can be computed as M{hypot_(*((c1 - c2) for c1, c2 in zip(p1, p2)))}, provided I{p1} and I{p2} have the same, non-zero length I{n}. ''' h, x2 = _h_x2(xs) return (h * sqrt(x2)) if x2 else _0_0 
 # In Python 3.8 and 3.9 C{math.hypot} is inaccurate, see # U{agdhruv<https://GitHub.com/geopy/geopy/issues/466>}, # U{cffk<https://Bugs.Python.org/issue43088>} and module # U{geomath.py<https://PyPI.org/project/geographiclib/1.52>} 
 '''Compute the norm M{sqrt(x**2 + y**2)}. 
 @arg x: X argument (C{scalar}). @arg y: Y argument (C{scalar}). 
 @return: C{sqrt(B{x}**2 + B{y}**2)} (C{float}). ''' else: 
 else: 
 
 '''(INTERNAL) Helper for L{hypot_} and L{hypot2_}. ''' 
 
 raise _ValueError(xs=xs, txt=_too_(_few_)) 
 
 '''Compute the norm M{sqrt(1 + x**2)}. 
 @arg x: Argument (C{scalar}). 
 @return: Norm (C{float}). ''' 
 
 '''Compute the I{squared} norm M{x**2 + y**2}. 
 @arg x: X argument (C{scalar}). @arg y: Y argument (C{scalar}). 
 @return: C{B{x}**2 + B{y}**2} (C{float}). ''' else: else: 
 
 '''Compute the I{squared} norm C{sum(x**2 for x in B{xs})}. 
 @arg xs: X arguments (C{scalar}s), all positional. 
 @return: Squared norm (C{float}). 
 @raise OverflowError: Partial C{2sum} overflow. 
 @raise ValueError: Invalid or no B{C{xs}} value. 
 @see: Function L{hypot_}. ''' 
 
 '''(INTERNAL) Yield B{C{a * b}}. ''' if len(a) != n: # PYCHOK no cover raise LenError(where, a=len(a), b=n) 
 
 '''(INTERNAL) Yield B{C{a * b}}, 1-primed. ''' 
 
 '''Normalize a 2-dimensional vector. 
 @arg x: X component (C{scalar}). @arg y: Y component (C{scalar}). 
 @return: 2-Tuple C{(x, y)}, normalized. 
 @raise ValueError: Invalid B{C{x}} or B{C{y}} or zero norm. ''' except Exception as e: raise _xError(e, x=x, y=y, h=h) 
 
 '''Normalize all n-dimensional vector components. 
 @arg xs: Components (C{scalar}s), all positional. 
 @return: Yield each component, normalized. 
 @raise ValueError: Invalid or insufficent B{C{xs}} or zero norm. ''' except Exception as e: raise _xError(e, Fmt.SQUARE(xs=i), x, _h_, h) 
 
 '''Compute the square root iff C{B{x2} >} L{EPS02}. 
 @arg x2: Value (C{scalar}). 
 @return: Square root (C{float}) or C{0.0}. 
 @note: Any C{B{x2} <} L{EPS02} I{including} C{B{x2} < 0} returns C{0.0}. ''' 
 
 '''Compute the square root, I{cubed} M{sqrt(x2)**3} or M{sqrt(x2**3)}. 
 @arg x2: Value (C{scalar}). 
 @return: Cubed square root (C{float}). 
 @raise ValueError: Negative B{C{x2}}. 
 @see: Functions L{cbrt} and L{cbrt2}. ''' raise _ValueError(x2=x2, txt=_negative_) 
 
 '''Compute C{I{a}} side of a right-angled triangle from C{sqrt(B{h}**2 - B{b}**2)}. 
 @arg h: Hypotenuse or outer annulus radius (C{scalar}). @arg b: Triangle side or inner annulus radius (C{scalar}). 
 @return: C{copysign(I{a}, B{h})} or C{unsigned 0.0} (C{float}). 
 @raise TypeError: Non-scalar B{C{h}} or B{C{b}}. 
 @raise ValueError: If C{abs(B{h}) < abs(B{b})}. 
 @see: Inner tangent chord B{I{d}} of an U{annulus <https://WikiPedia.org/wiki/Annulus_(mathematics)>} and function U{annulus_area<https://People.SC.FSU.edu/ ~jburkardt/py_src/geometry/geometry.py>}. ''' raise TypeError(_not_scalar_) raise ValueError('abs(h) < abs(b)') 
 if isnear0(h): # PYCHOK no cover c, b = fabs(h), fabs(b) d = c - b a = copysign0(sqrt((c + b) * d), h) if d > 0 else _0_0 else: 
 except Exception as x: raise _xError(x, h=h, b=b) 
 
 # **) MIT License # # Copyright (C) 2016-2022 -- mrJean1 at Gmail -- All Rights Reserved. # # Permission is hereby granted, free of charge, to any person obtaining a # copy of this software and associated documentation files (the "Software"), # to deal in the Software without restriction, including without limitation # the rights to use, copy, modify, merge, publish, distribute, sublicense, # and/or sell copies of the Software, and to permit persons to whom the # Software is furnished to do so, subject to the following conditions: # # The above copyright notice and this permission notice shall be included # in all copies or substantial portions of the Software. # # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS # OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL # THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR # OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, # ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR # OTHER DEALINGS IN THE SOFTWARE. |