-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathmath.go
156 lines (135 loc) · 3.97 KB
/
math.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
package gg
import (
"math"
)
/*
Short for "is finite". Missing feature of the standard "math" package.
True if the input is neither NaN nor infinity.
*/
func IsFin[A Float](val A) bool {
tar := float64(val)
return !math.IsNaN(tar) && !math.IsInf(tar, 0)
}
// Short for "is natural". True if >= 0. Also see `IsPos`.
func IsNat[A Num](val A) bool { return val >= 0 }
// Short for "is positive". True if > 0. Also see `IsNat`.
func IsPos[A Num](val A) bool { return val > 0 }
// Short for "is negative". True if < 0. Also see `IsNat`.
func IsNeg[A Num](val A) bool { return val < 0 }
/*
True if the remainder of dividing the first argument by the second argument is
zero. If the divisor is zero, does not attempt the division and returns false.
Note that the result is unaffected by the signs of either the dividend or the
divisor.
*/
func IsDivisibleBy[A Int](dividend, divisor A) bool {
return divisor != 0 && dividend%divisor == 0
}
// True if the input has a fractional component.
func IsFrac[A Float](val A) bool {
_, frac := math.Modf(float64(val))
return frac != 0 && !math.IsNaN(frac)
}
// Same as `Add(val, 1)`. Panics on overflow.
func Inc[A Int](val A) A { return Add(val, 1) }
// Same as `Sub(val, 1)`. Panics on underflow.
func Dec[A Int](val A) A { return Sub(val, 1) }
/*
Raises a number to a power. Same as `math.Pow` and calls it under the hood, but
accepts arbitrary numeric types and performs checked conversions via `NumConv`.
Panics on overflow or precision loss. Has minor overhead over `math.Pow`.
Compare `PowUncheck` which runs faster but may overflow.
*/
func Pow[Tar, Pow Num](src Tar, pow Pow) Tar {
return NumConv[Tar](math.Pow(NumConv[float64](src), NumConv[float64](pow)))
}
/*
Raises a number to a power. Same as `math.Pow` and calls it under the hood, but
accepts arbitrary numeric types. Does not check for overflow or precision loss.
Counterpart to `Pow` which panics on overflow.
*/
func PowUncheck[Tar, Pow Num](src Tar, pow Pow) Tar {
return Tar(math.Pow(float64(src), float64(pow)))
}
/*
Checked factorial. Panics on overflow. Compare `FacUncheck` which runs faster,
but may overflow.
*/
func Fac[A Uint](src A) A {
var tar float64 = 1
mul := NumConv[float64](src)
for mul > 0 {
tar *= mul
mul--
}
return NumConv[A](tar)
}
/*
Unchecked factorial. May overflow. Counterpart to `Fac` which panics on
overflow.
*/
func FacUncheck[A Uint](src A) A {
var out A = 1
for src > 0 {
out *= src
src -= 1
}
return out
}
// Checked addition. Panics on overflow/underflow. Has overhead.
func Add[A Int](one, two A) A {
out := one + two
if (out > one) == (two > 0) {
return out
}
panic(errAdd(one, two, out))
}
func errAdd[A Int](one, two, out A) Err {
return Errf(
`addition overflow for %v: %v + %v = %v`,
Type[A](), one, two, out,
)
}
/*
Unchecked addition. Same as Go's `+` operator for numbers, expressed as a
generic function. Does not take strings. May overflow. For integers, prefer
`Add` whenever possible, which has overflow checks.
*/
func AddUncheck[A Num](one, two A) A { return one + two }
// Checked subtraction. Panics on overflow/underflow. Has overhead.
func Sub[A Int](one, two A) A {
out := one - two
if (out < one) == (two > 0) {
return out
}
panic(errSub(one, two, out))
}
func errSub[A Int](one, two, out A) Err {
return Errf(
`subtraction overflow for %v: %v - %v = %v`,
Type[A](), one, two, out,
)
}
/*
Unchecked subtraction. Same as Go's `-` operator, expressed as a generic
function. May overflow. For integers, prefer `Sub` whenever possible, which has
overflow checks.
*/
func SubUncheck[A Num](one, two A) A { return one - two }
// Checked multiplication. Panics on overflow/underflow. Has overhead.
func Mul[A Int](one, two A) A {
if one == 0 || two == 0 {
return 0
}
out := one * two
if ((one < 0) == (two < 0)) != (out < 0) && out/two == one {
return out
}
panic(errMul(one, two, out))
}
func errMul[A Int](one, two, out A) Err {
return Errf(
`multiplication overflow for %v: %v * %v = %v`,
Type[A](), one, two, out,
)
}