Module Core.FloatSource
Floating-point numbers.
include Core.Identifiable.S__local
with type t := Base.Float.t
and type comparator_witness := Base.Float.comparator_witness
include Sexplib0.Sexpable.S__stack with type t := Base.Float.t
val sexp_of_t__stack : Base.Float.t @ local -> Sexplib0.Sexp.t @ localval t_sexp_grammar : Base.Float.t Sexplib0.Sexp_grammar.t @@ portableinclude Base.Floatable.S with type t := Base.Float.t
val of_float : float -> Base.Float.tval to_float : Base.Float.t -> floatmax and min will return nan if either argument is nan.
include Base.Identifiable.S__local__portable
with type t := Base.Float.t
with type comparator_witness = Base.Float.comparator_witness
val hash_fold_t : Base.Float.t Ppx_hash_lib.hash_foldval hash : Base.Float.t -> Ppx_hash_lib.Std.Hash.hash_valueval equal__local :
Base.Float.t @ local ->
Base.Float.t @ local ->
bool @@ portableval compare__local :
Base.Float.t @ local ->
Base.Float.t @ local ->
int @@ portableval ascending : Base.Float.t -> Base.Float.t -> int @@ portableval descending : Base.Float.t -> Base.Float.t -> int @@ portableval between :
Base.Float.t ->
low:Base.Float.t ->
high:Base.Float.t ->
bool @@ portableval clamp_exn :
Base.Float.t ->
min:Base.Float.t ->
max:Base.Float.t ->
Base.Float.t @@ portableval clamp :
Base.Float.t ->
min:Base.Float.t ->
max:Base.Float.t ->
Base.Float.t Base.Or_error.t @@ portabletype comparator_witness = Base.Float.comparator_witnessval comparator :
(Base.Float.t, Core.Float.comparator_witness) Base__Comparator.t @@ portableval pp : Base.Formatter.t -> Base.Float.t -> unitval hashable : Base.Float.t Base.Hashable.tval of_string_opt : string @ local -> Base.Float.t optioninclude Base.Comparable.With_zero__local with type t := Base.Float.t
val is_positive : Base.Float.t @ local -> boolval is_non_negative : Base.Float.t @ local -> boolval is_negative : Base.Float.t @ local -> boolval is_non_positive : Base.Float.t @ local -> boolinclude Base.Invariant.S with type t := Base.Float.t
val invariant : Base.Float.t -> unitinclude Base.Comparisons.S_with_local_opt with type t := Base.Float.t
include Base.Comparisons.Infix_with_local_opt with type t := Base.Float.t
val (<) : Base.Float.t -> Base.Float.t -> boolval (<=) : Base.Float.t -> Base.Float.t -> boolval (<>) : Base.Float.t -> Base.Float.t -> boolval (=) : Base.Float.t -> Base.Float.t -> boolval (>) : Base.Float.t -> Base.Float.t -> boolval (>=) : Base.Float.t -> Base.Float.t -> boolinclude sig ... end
val equal : Base.Float.t -> Base.Float.t -> boolval compare : Base.Float.t -> Base.Float.t -> intinclude sig ... end
val min : Base.Float.t -> Base.Float.t -> Base.Float.tval max : Base.Float.t -> Base.Float.t -> Base.Float.tval nan : Base.Float.tval infinity : Base.Float.tval neg_infinity : Base.Float.tval max_value : Base.Float.tEqual to infinity.
val min_value : Base.Float.tEqual to neg_infinity.
val zero : Base.Float.tval one : Base.Float.tval minus_one : Base.Float.tval pi : Base.Float.tThe constant pi.
val sqrt_pi : Base.Float.tThe constant sqrt(pi).
val sqrt_2pi : Base.Float.tThe constant sqrt(2 * pi).
val euler_gamma_constant : Base.Float.tEuler-Mascheroni constant (γ).
val epsilon_float : Base.Float.tThe difference between 1.0 and the smallest exactly representable floating-point number greater than 1.0. That is:
epsilon_float = (one_ulp `Up 1.0) -. 1.0
This gives the relative accuracy of type t, in the sense that for numbers on the order of x, the roundoff error is on the order of x *. float_epsilon.
See also: Machine epsilon.
val max_finite_value : Base.Float.tmin_positive_subnormal_value = 2 ** -1074min_positive_normal_value = 2 ** -1022
val min_positive_subnormal_value : Base.Float.tval min_positive_normal_value : Base.Float.tval to_int64_preserve_order : Base.Float.t @ local -> int64 optionAn order-preserving bijection between all floats except for nans, and all int64s with absolute value smaller than or equal to 2**63 - 2**52. Note both 0. and -0. map to 0L.
val to_int64_preserve_order_exn : Base.Float.t @ local -> int64val of_int64_preserve_order : int64 @ local -> Base.Float.tReturns nan if the absolute value of the argument is too large.
val one_ulp : [ `Up | `Down ] -> Base.Float.t @ local -> Base.Float.tThe next or previous representable float. ULP stands for "unit of least precision", and is the spacing between floating point numbers. Both one_ulp `Up infinity and one_ulp `Down neg_infinity return a nan.
val of_int : int -> Base.Float.tNote that this doesn't round trip in either direction. For example, Float.to_int (Float.of_int max_int) <> max_int.
val to_int : Base.Float.t @ local -> intval of_int63 : Base.Int63.t @ local -> Base.Float.tval of_int64 : int64 @ local -> Base.Float.tval to_int64 : Base.Float.t @ local -> int64val round :
?dir:[ `Zero | `Nearest | `Up | `Down ] @ local ->
Base.Float.t @ local ->
Base.Float.tround rounds a float to an integer float. iround{,_exn} rounds a float to an int. Both round according to a direction dir, with default dir being `Nearest.
| `Down | rounds toward Float.neg_infinity |
| `Up | rounds toward Float.infinity |
| `Nearest | rounds to the nearest int ("round half-integers up") |
| `Zero | rounds toward zero |iround_exn raises when trying to handle nan or trying to handle a float outside the range [float min_int, float max_int).
Here are some examples for round for each direction:
| `Down | [-2.,-1.) to -2. | [-1.,0.) to -1. | [0.,1.) to 0., [1.,2.) to 1. | | `Up | (-2.,-1.] to -1. | (-1.,0.] to -0. | (0.,1.] to 1., (1.,2.] to 2. | | `Zero | (-2.,-1.] to -1. | (-1.,1.) to 0. | [1.,2.) to 1. | | `Nearest | [-1.5,-0.5) to -1. | [-0.5,0.5) to 0. | [0.5,1.5) to 1. |
For convenience, versions of these functions with the dir argument hard-coded are provided. If you are writing performance-critical code you should use the versions with the hard-coded arguments (e.g. iround_down_exn). The _exn ones are the fastest.
The following properties hold:
of_int (iround_*_exn i) = ifor any floatithat is an integer withmin_int <= i <= max_int.
round_* i = ifor any floatithat is an integer.
iround_*_exn (of_int i) = ifor any intiwith-2**52 <= i <= 2**52.
val iround :
?dir:[ `Zero | `Nearest | `Up | `Down ] @ local ->
Base.Float.t @ local ->
int optionval iround_exn :
?dir:[ `Zero | `Nearest | `Up | `Down ] @ local ->
Base.Float.t @ local ->
intval round_towards_zero : Base.Float.t @ local -> Base.Float.tval round_down : Base.Float.t @ local -> Base.Float.tval round_up : Base.Float.t @ local -> Base.Float.tval round_nearest : Base.Float.t @ local -> Base.Float.tRounds half integers up.
val round_nearest_half_to_even : Base.Float.t @ local -> Base.Float.tRounds half integers to the even integer.
val iround_towards_zero : Base.Float.t @ local -> int optionval iround_down : Base.Float.t @ local -> int optionval iround_up : Base.Float.t @ local -> int optionval iround_nearest : Base.Float.t @ local -> int optionval iround_towards_zero_exn : Base.Float.t @ local -> intval iround_down_exn : Base.Float.t @ local -> intval iround_up_exn : Base.Float.t @ local -> intval iround_nearest_exn : Base.Float.t @ local -> intval int63_round_down_exn : Base.Float.t @ local -> Base.Int63.tval int63_round_up_exn : Base.Float.t @ local -> Base.Int63.tval int63_round_nearest_exn : Base.Float.t @ local -> Base.Int63.tval iround_lbound : Base.Float.tIf f < iround_lbound || f > iround_ubound, then iround* functions will refuse to round f, returning None or raising as appropriate.
val iround_ubound : Base.Float.tval int63_round_lbound : Base.Float.tval int63_round_ubound : Base.Float.tval round_significant : Base.Float.t -> significant_digits:int -> Base.Float.tround_significant x ~significant_digits:n rounds to the nearest number with n significant digits. More precisely: it returns the representable float closest to x rounded to n significant digits. It is meant to be equivalent to sprintf "%.*g" n x |> Float.of_string but faster (10x-15x). Exact ties are resolved as round-to-even.
However, it might in rare cases break the contract above.
It might in some cases appear as if it violates the round-to-even rule:
let x = 4.36083208835
let z = 4.3608320883;;
assert (z = fast_approx_round_significant x ~sf:11)But in this case so does sprintf, since x as a float is slightly under-represented:
sprintf "%.11g" x = "4.3608320883";;
sprintf "%.30g" x = "4.36083208834999958014577714493"More importantly, round_significant might sometimes give a different result than sprintf ... |> Float.of_string because it round-trips through an integer. For example, the decimal fraction 0.009375 is slightly under-represented as a float:
sprintf "%.17g" 0.009375 = "0.0093749999999999997"But:
0.009375 *. 1e5 = 937.5Therefore:
round_significant 0.009375 ~significant_digits:3 = 0.00938whereas:
sprintf "%.3g" 0.009375 = "0.00937"In general we believe (and have tested on numerous examples) that the following holds for all x:
let s = sprintf "%.*g" significant_digits x |> Float.of_string in
s = round_significant ~significant_digits x
|| s = round_significant ~significant_digits (one_ulp `Up x)
|| s = round_significant ~significant_digits (one_ulp `Down x)Also, for float representations of decimal fractions (like 0.009375), round_significant is more likely to give the "desired" result than sprintf ... |> of_string (that is, the result of rounding the decimal fraction, rather than its float representation). But it's not guaranteed either--see the 4.36083208835 example above.
val round_decimal : Base.Float.t -> decimal_digits:int -> Base.Float.tround_decimal x ~decimal_digits:n rounds x to the nearest 10**(-n). For positive n it is meant to be equivalent to sprintf "%.*f" n x |> Float.of_string, but faster.
All the considerations mentioned in round_significant apply (both functions use the same code path).
val is_nan : Base.Float.t @ local -> boolval is_inf : Base.Float.t @ local -> boolA float is infinite when it is either infinity or neg_infinity.
val is_finite : Base.Float.t @ local -> boolA float is finite when neither is_nan nor is_inf is true.
val is_integer : Base.Float.t @ local -> boolis_integer x is true if and only if x is an integer.
min_inan and max_inan return, respectively, the min and max of the two given values, except when one of the values is a nan, in which case the other is returned. (Returns nan if both arguments are nan.)
include sig ... end
val min_inan : Base.Float.t -> Base.Float.t -> Base.Float.tval max_inan : Base.Float.t -> Base.Float.t -> Base.Float.tval (+) : Base.Float.t -> Base.Float.t -> Base.Float.tval (-) : Base.Float.t -> Base.Float.t -> Base.Float.tval (*) : Base.Float.t -> Base.Float.t -> Base.Float.tval (/) : Base.Float.t -> Base.Float.t -> Base.Float.tval (%) : Base.Float.t -> Base.Float.t -> Base.Float.tIn analogy to Int.( % ), ( % ):
- always produces non-negative (or NaN) result
- raises when given a negative modulus.
Like the other infix operators, NaNs in mean NaNs out.
Other cases: (a % Infinity) = a when 0 <= a < Infinity, (a % Infinity) = Infinity when -Infinity < a < 0, (+/- Infinity % a) = NaN, (a % 0) = NaN.
val (**) : Base.Float.t -> Base.Float.t -> Base.Float.tval (~-) : Base.Float.t -> Base.Float.tReturns the fractional part and the whole (i.e., integer) part. For example, modf (-3.14) returns { fractional = -0.14; integral = -3.; }!
val modf : Base.Float.t -> Core.Float.Parts.tval mod_float : Base.Float.t -> Base.Float.t -> Base.Float.tmod_float x y returns a result with the same sign as x. It returns nan if y is 0. It is basically
let mod_float x y = x -. (float (truncate (x /. y)) *. y)not
let mod_float x y = x -. (floor (x /. y) *. y)and therefore resembles mod on integers more than %.
val add : Base.Float.t -> Base.Float.t -> Base.Float.tOrdinary functions for arithmetic operations
These are for modules that inherit from t, since the infix operators are more convenient.
val sub : Base.Float.t -> Base.Float.t -> Base.Float.tval neg : Base.Float.t -> Base.Float.tval scale : Base.Float.t -> Base.Float.t -> Base.Float.tval abs : Base.Float.t -> Base.Float.tSimilar to O, except that operators are suffixed with a dot, allowing one to have both int and float operators in scope simultaneously.
val to_string_hum :
?delimiter:char ->
?decimals:int ->
?strip_zero:bool ->
?explicit_plus:bool ->
Base.Float.t @ local ->
stringPretty print float, for example to_string_hum ~decimals:3 1234.1999 = "1_234.200" to_string_hum ~decimals:3 ~strip_zero:true 1234.1999 = "1_234.2" . No delimiters are inserted to the right of the decimal.
val to_padded_compact_string : Base.Float.t @ local -> stringProduce a lossy compact string representation of the float. The float is scaled by an appropriate power of 1000 and rendered with one digit after the decimal point, except that the decimal point is written as '.', 'k', 'm', 'g', 't', or 'p' to indicate the scale factor. (However, if the digit after the "decimal" point is 0, it is suppressed.)
The smallest scale factor that allows the number to be rendered with at most 3 digits to the left of the decimal is used. If the number is too large for this format (i.e., the absolute value is at least 999.95e15), scientific notation is used instead. E.g.:
to_padded_compact_string (-0.01) = "-0 "to_padded_compact_string 1.89 = "1.9"to_padded_compact_string 999_949.99 = "999k9"to_padded_compact_string 999_950. = "1m "
In the case where the digit after the "decimal", or the "decimal" itself is omitted, the numbers are padded on the right with spaces to ensure the last two columns of the string always correspond to the decimal and the digit afterward (except in the case of scientific notation, where the exponent is the right-most element in the string and could take up to four characters).
to_padded_compact_string 1. = "1 "to_padded_compact_string 1.e6 = "1m "to_padded_compact_string 1.e16 = "1.e+16"to_padded_compact_string max_finite_value = "1.8e+308"
Numbers in the range -.05 < x < .05 are rendered as "0 " or "-0 ".
Other cases:
to_padded_compact_string nan = "nan "to_padded_compact_string infinity = "inf "to_padded_compact_string neg_infinity = "-inf "
Exact ties are resolved to even in the decimal:
to_padded_compact_string 3.25 = "3.2"to_padded_compact_string 3.75 = "3.8"to_padded_compact_string 33_250. = "33k2"to_padded_compact_string 33_350. = "33k4"
to_padded_compact_string is defined in terms of to_padded_compact_string_custom below as
let to_padded_compact_string t =
to_padded_compact_string_custom
t
?prefix:None
~kilo:"k"
~mega:"m"
~giga:"g"
~tera:"t"
~peta:"p"
()
;;val to_padded_compact_string_custom :
Base.Float.t @ local ->
?prefix:string ->
kilo:string ->
mega:string ->
giga:string ->
tera:string ->
?peta:string ->
unit ->
stringSimilar to to_padded_compact_string but allows the user to provide different abbreviations. This can be useful to display currency values, e.g. $1mm3, where prefix="$", mega="mm".
val int_pow : Base.Float.t @ local -> int -> Base.Float.tint_pow x n computes x ** float n via repeated squaring. It is generally much faster than **.
Note that int_pow x 0 always returns 1., even if x = nan. This coincides with x ** 0. and is intentional.
For n >= 0 the result is identical to an n-fold product of x with itself under *., with a certain placement of parentheses. For n < 0 the result is identical to int_pow (1. /. x) (-n).
The error will be on the order of |n| ulps, essentially the same as if you perturbed x by up to a ulp and then exponentiated exactly.
Benchmarks show a factor of 5-10 speedup (relative to **) for exponents up to about 1000 (approximately 10ns vs. 70ns). For larger exponents the advantage is smaller but persists into the trillions. For a recent or more detailed comparison, run the benchmarks.
Depending on context, calling this function might or might not allocate 2 minor words. Even if called in a way that causes allocation, it still appears to be faster than **.
val square : Base.Float.t -> Base.Float.tsquare x returns x *. x.
val ldexp : Base.Float.t -> int -> Base.Float.tldexp x n returns x *. 2 ** n
val frexp : Base.Float.t -> Base.Float.t * intfrexp f returns the pair of the significant and the exponent of f. When f is zero, the significant x and the exponent n of f are equal to zero. When f is non-zero, they are defined by f = x *. 2 ** n and 0.5 <= x < 1.0.
val log10 : Base.Float.t -> Base.Float.tBase 10 logarithm.
val log2 : Base.Float.t -> Base.Float.tBase 2 logarithm.
val expm1 : Base.Float.t -> Base.Float.texpm1 x computes exp x -. 1.0, giving numerically-accurate results even if x is close to 0.0.
val log1p : Base.Float.t -> Base.Float.tlog1p x computes log(1.0 +. x) (natural logarithm), giving numerically-accurate results even if x is close to 0.0.
val copysign : Base.Float.t -> Base.Float.t -> Base.Float.tcopysign x y returns a float whose absolute value is that of x and whose sign is that of y. If x is nan, returns nan. If y is nan, returns either x or -. x, but it is not specified which.
val cos : Base.Float.t -> Base.Float.tCosine. Argument is in radians.
val sin : Base.Float.t -> Base.Float.tSine. Argument is in radians.
val tan : Base.Float.t -> Base.Float.tTangent. Argument is in radians.
val acos : Base.Float.t -> Base.Float.tArc cosine. The argument must fall within the range [-1.0, 1.0]. Result is in radians and is between 0.0 and pi.
val asin : Base.Float.t -> Base.Float.tArc sine. The argument must fall within the range [-1.0, 1.0]. Result is in radians and is between -pi/2 and pi/2.
val atan : Base.Float.t -> Base.Float.tArc tangent. Result is in radians and is between -pi/2 and pi/2.
val atan2 : Base.Float.t -> Base.Float.t -> Base.Float.tatan2 y x returns the arc tangent of y /. x. The signs of x and y are used to determine the quadrant of the result. Result is in radians and is between -pi and pi.
val hypot : Base.Float.t -> Base.Float.t -> Base.Float.thypot x y returns sqrt(x *. x + y *. y), that is, the length of the hypotenuse of a right-angled triangle with sides of length x and y, or, equivalently, the distance of the point (x,y) to origin.
val cosh : Base.Float.t -> Base.Float.tHyperbolic cosine. Argument is in radians.
val sinh : Base.Float.t -> Base.Float.tHyperbolic sine. Argument is in radians.
val tanh : Base.Float.t -> Base.Float.tHyperbolic tangent. Argument is in radians.
val acosh : Base.Float.t -> Base.Float.tHyperbolic arc cosine. The argument must fall within the range [1.0, inf]. Result is in radians and is between 0.0 and inf.
val asinh : Base.Float.t -> Base.Float.tHyperbolic arc sine. The argument and result range over the entire real line. Result is in radians.
val atanh : Base.Float.t -> Base.Float.tHyperbolic arc tangent. The argument must fall within the range [-1.0, 1.0]. Result is in radians and ranges over the entire real line.
val sqrt : Base.Float.t -> Base.Float.tSquare root.
val exp : Base.Float.t -> Base.Float.tExponential.
val log : Base.Float.t -> Base.Float.tNatural logarithm.
val classify : Base.Float.t @ local -> Core.Float.Class.tval sign_exn : Base.Float.t @ local -> Base.Sign.tThe sign of a float. Both -0. and 0. map to Zero. Raises on nan. All other values map to Neg or Pos.
val sign_or_nan : Base.Float.t @ local -> Base.Sign_or_nan.tThe sign of a float, with support for NaN. Both -0. and 0. map to Zero. All NaN values map to Nan. All other values map to Neg or Pos.
val create_ieee :
negative:bool ->
exponent:int ->
mantissa:Base.Int63.t @ local ->
Base.Float.t Base.Or_error.tThese functions construct and destruct 64-bit floating point numbers based on their IEEE representation with a sign bit, an 11-bit non-negative (biased) exponent, and a 52-bit non-negative mantissa (or significand). See Wikipedia for details of the encoding.
In particular, if 1 <= exponent <= 2046, then:
create_ieee_exn ~negative:false ~exponent ~mantissa
= (2 ** (exponent - 1023)) * (1 + ((2 ** -52) * mantissa))val create_ieee_exn :
negative:bool ->
exponent:int ->
mantissa:Base.Int63.t @ local ->
Base.Float.tval ieee_negative : Base.Float.t @ local -> boolval ieee_exponent : Base.Float.t @ local -> intval ieee_mantissa : Base.Float.t @ local -> Base.Int63.tinclude Bin_prot.Binable.S__local with type t := Core.Float.t
This function only needs implementation if t exposed to be a polymorphic variant. Despite what the type reads, this does *not* produce a function after reading; instead it takes the constructor tag (int) before reading and reads the rest of the variant t afterwards.
So-called "robust" comparisons, which include a small tolerance, so that float that differ by a small amount are considered equal.
Note that the results of robust comparisons on nan should be considered undefined.
include Core.Float.Robust_compare.S
intended to be a tolerance on human-entered floats
validate_lbound, validate_ubound, and validate_bound always fail if class is Nan or Infinite. The other validation functions still fail on Nan, but permit Infinite values of the correct sign. (The behavior with respect to infinity will probably be changed to be more consistent.)
include sig ... end
val validate_bound :
min:Core.Float.t Core.Maybe_bound.t ->
max:Core.Float.t Core.Maybe_bound.t ->
Core.Float.t Validate.checkinclude sig ... end
validate_ordinary fails if class is Nan or Infinite.
to_string_12 x builds a string representing x using up to 12 significant digits. It loses precision. You can use "%{Float#12}" in formats, but consider "%.12g", "%{Float#hum}", or "%{Float}" as alternatives.
to_string x builds a string s representing the float x that guarantees the round trip, i.e., Float.equal x (Float.of_string s).
It usually yields as few significant digits as possible. That is, it won't print 3.14 as 3.1400000000000001243. The only exception is that occasionally it will output 17 significant digits when the number can be represented with just 16 (but not 15 or fewer) of them.
include Base.Stringable.S_local_input with type t := Core.Float.t
val of_string : string @ local -> Core.Float.tval to_string : Core.Float.t @ local -> string(Formerly sign) Uses robust comparison (so sufficiently small numbers are mapped to Zero). Also maps NaN to Zero. Using this function is weakly discouraged.
val gen_uniform_excl :
Core.Float.t ->
Core.Float.t ->
Core.Float.t Core.Quickcheck.Generator.t @ portable @@ portablegen_uniform_excl lo hi creates a Quickcheck generator producing finite t values between lo and hi, exclusive. The generator approximates a uniform distribution over the interval (lo, hi). Raises an exception if lo is not finite, hi is not finite, or the requested range is empty.
The implementation chooses values uniformly distributed between 0 (inclusive) and 1 (exclusive) up to 52 bits of precision, then scales that interval to the requested range. Due to rounding errors and non-uniform floating point precision, the resulting distribution may not be precisely uniform and may not include all values between lo and hi.
val gen_incl :
Core.Float.t ->
Core.Float.t ->
Core.Float.t Core.Quickcheck.Generator.t @ portable @@ portablegen_incl lo hi creates a Quickcheck generator that produces values between lo and hi, inclusive, approximately uniformly distributed, with extra weight given to generating the endpoints lo and hi. Raises an exception if lo is not finite, hi is not finite, or the requested range is empty.
gen_finite produces all finite t values, excluding infinities and all NaN values.
gen_positive produces all (strictly) positive finite t values.
gen_negative produces all (strictly) negative finite t values.
gen_without_nan produces all finite and infinite t values, excluding all NaN values.
gen_infinite produces both infinite values
gen_nan produces all NaN values.
gen_normal produces all normal values
gen_subnormal produces all subnormal values
gen_zero produces both zero values