spire.std.float.scala Maven / Gradle / Ivy
package spire.std
import spire.algebra.{Field, IsRational, NRoot, Order, Signed, Trig}
import spire.math.Rational
import java.lang.Math
import java.lang.Integer.{ numberOfTrailingZeros, numberOfLeadingZeros }
import java.lang.Float.{ intBitsToFloat, floatToIntBits }
import scala.annotation.tailrec
trait FloatIsField extends Field[Float] {
override def minus(a:Float, b:Float): Float = a - b
def negate(a:Float): Float = -a
def one: Float = 1.0F
def plus(a:Float, b:Float): Float = a + b
override def pow(a:Float, b:Int): Float = Math.pow(a, b).toFloat
override def times(a:Float, b:Float): Float = a * b
def zero: Float = 0.0F
override def fromInt(n: Int): Float = n
def quot(a:Float, b:Float): Float = (a - (a % b)) / b
def mod(a:Float, b:Float): Float = a % b
final def gcd(a:Float, b:Float):Float = {
def value(bits: Int): Int = bits & 0x007FFFFF | 0x00800000
def exp(bits: Int): Int = ((bits >> 23) & 0xFF).toInt
// Computes the GCD of 2 fp values. Here, we are guaranteed that exp0 < exp1.
def gcd0(val0: Int, exp0: Int, val1: Int, exp1: Int): Float = {
val tz0 = numberOfTrailingZeros(val0)
val tz1 = numberOfTrailingZeros(val1)
val tzShared = spire.math.min(tz0, tz1 + exp1 - exp0)
// We trim of the power of 2s, then add back the shared portion.
val n = spire.math.gcd(val0 >>> tz0, val1 >>> tz1).toInt << tzShared
// Number of bits to move the leading 1 to bit position 23.
val shift = numberOfLeadingZeros(n) - 8
val exp = (exp0 - shift)
// If exp is 0, then the value is actually just the mantissa * 2^−126,
// so we need to adjust the *shift* accordingly.
val shift0 = if (exp == 0) shift - 1 else shift
val mantissa = (n << shift0) & 0x007FFFFF
// If exp < 0, then we have underflowed; not much we can do but return 0.
if (exp < 0) 0F
else intBitsToFloat((exp << 23) | mantissa)
}
if (a == 0F) b
else if (b == 0F) a
else {
val aBits = floatToIntBits(a)
val aVal = value(aBits)
val aExp = exp(aBits)
val bBits = floatToIntBits(b)
val bVal = value(bBits)
val bExp = exp(bBits)
if (aExp < bExp) gcd0(aVal, aExp, bVal, bExp)
else gcd0(bVal, bExp, aVal, aExp)
}
}
override def fromDouble(n: Double): Float = n.toFloat
def div(a:Float, b:Float): Float = a / b
}
trait FloatIsNRoot extends NRoot[Float] {
def nroot(a: Float, k: Int): Float = Math.pow(a, 1 / k.toDouble).toFloat
override def sqrt(a: Float): Float = Math.sqrt(a).toFloat
def fpow(a: Float, b: Float): Float = Math.pow(a, b).toFloat
}
trait FloatIsTrig extends Trig[Float] {
def e: Float = Math.E.toFloat
def pi: Float = Math.PI.toFloat
def exp(a: Float): Float = Math.exp(a).toFloat
def expm1(a: Float): Float = Math.expm1(a).toFloat
def log(a: Float): Float = Math.log(a).toFloat
def log1p(a: Float): Float = Math.log1p(a).toFloat
def sin(a: Float): Float = Math.sin(a.toDouble).toFloat
def cos(a: Float): Float = Math.cos(a.toDouble).toFloat
def tan(a: Float): Float = Math.tan(a.toDouble).toFloat
def asin(a: Float): Float = Math.asin(a.toDouble).toFloat
def acos(a: Float): Float = Math.acos(a.toDouble).toFloat
def atan(a: Float): Float = Math.atan(a.toDouble).toFloat
def atan2(y: Float, x: Float): Float = Math.atan2(y.toDouble, x.toDouble).toFloat
def sinh(x: Float): Float = Math.sinh(x.toDouble).toFloat
def cosh(x: Float): Float = Math.cosh(x.toDouble).toFloat
def tanh(x: Float): Float = Math.tanh(x.toDouble).toFloat
def toRadians(a: Float): Float = (a * 2 * pi) / 360
def toDegrees(a: Float): Float = (a * 360) / (2 * pi)
}
trait FloatIsSigned extends Signed[Float] {
def signum(a: Float): Int = Math.signum(a).toInt
def abs(a: Float): Float = if (a < 0.0f) -a else a
}
trait FloatOrder extends Order[Float] {
override def eqv(x:Float, y:Float): Boolean = x == y
override def neqv(x:Float, y:Float): Boolean = x != y
override def gt(x: Float, y: Float): Boolean = x > y
override def gteqv(x: Float, y: Float): Boolean = x >= y
override def lt(x: Float, y: Float): Boolean = x < y
override def lteqv(x: Float, y: Float): Boolean = x <= y
override def min(x: Float, y: Float): Float = Math.min(x, y)
override def max(x: Float, y: Float): Float = Math.max(x, y)
def compare(x: Float, y: Float): Int = java.lang.Float.compare(x, y)
}
trait FloatIsReal extends IsRational[Float] with FloatOrder with FloatIsSigned {
def toDouble(x: Float): Double = x.toDouble
def ceil(a:Float): Float = Math.ceil(a).toFloat
def floor(a:Float): Float = Math.floor(a).toFloat
def round(a:Float): Float = spire.math.round(a)
def isWhole(a:Float): Boolean = a % 1.0 == 0.0
def toRational(a:Float): Rational = Rational(a)
}
@SerialVersionUID(0L)
class FloatAlgebra extends FloatIsField with FloatIsNRoot with FloatIsTrig with FloatIsReal with Serializable
trait FloatInstances {
implicit final val FloatAlgebra = new FloatAlgebra
import Float._
import spire.math.NumberTag._
implicit final val FloatTag = new BuiltinFloatTag(0F, MinValue, MaxValue, NaN, PositiveInfinity, NegativeInfinity) {
def isInfinite(a: Float): Boolean = java.lang.Float.isInfinite(a)
def isNaN(a: Float): Boolean = java.lang.Float.isNaN(a)
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy