bigdecimal 4.0.1 → 4.1.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/bigdecimal.gemspec +6 -1
- data/ext/bigdecimal/bigdecimal.c +262 -261
- data/ext/bigdecimal/bigdecimal.h +43 -37
- data/ext/bigdecimal/div.h +192 -0
- data/ext/bigdecimal/extconf.rb +5 -2
- data/ext/bigdecimal/missing/dtoa.c +184 -137
- data/ext/bigdecimal/missing.h +4 -2
- data/ext/bigdecimal/ntt.h +191 -0
- data/lib/bigdecimal/math.rb +102 -123
- data/lib/bigdecimal/util.rb +1 -1
- data/lib/bigdecimal.rb +104 -60
- data/sample/linear.rb +73 -37
- data/sample/nlsolve.rb +47 -30
- data/sample/pi.rb +2 -7
- data/sig/big_decimal.rbs +1502 -0
- data/sig/big_decimal_util.rbs +158 -0
- data/sig/big_math.rbs +423 -0
- metadata +7 -2
data/lib/bigdecimal/util.rb
CHANGED
data/lib/bigdecimal.rb
CHANGED
|
@@ -12,6 +12,9 @@ end
|
|
|
12
12
|
|
|
13
13
|
class BigDecimal
|
|
14
14
|
module Internal # :nodoc:
|
|
15
|
+
# Default extra precision for intermediate calculations
|
|
16
|
+
# This value is currently the same as BigDecimal.double_fig, but defined separately for future changes.
|
|
17
|
+
EXTRA_PREC = 16
|
|
15
18
|
|
|
16
19
|
# Coerce x to BigDecimal with the specified precision.
|
|
17
20
|
# TODO: some methods (example: BigMath.exp) require more precision than specified to coerce.
|
|
@@ -54,12 +57,59 @@ class BigDecimal
|
|
|
54
57
|
BigDecimal::INFINITY
|
|
55
58
|
end
|
|
56
59
|
|
|
60
|
+
def self.underflow_computation_result # :nodoc:
|
|
61
|
+
if BigDecimal.mode(BigDecimal::EXCEPTION_ALL).anybits?(BigDecimal::EXCEPTION_UNDERFLOW)
|
|
62
|
+
raise FloatDomainError, 'Exponent underflow'
|
|
63
|
+
end
|
|
64
|
+
BigDecimal(0)
|
|
65
|
+
end
|
|
66
|
+
|
|
57
67
|
def self.nan_computation_result # :nodoc:
|
|
58
68
|
if BigDecimal.mode(BigDecimal::EXCEPTION_ALL).anybits?(BigDecimal::EXCEPTION_NaN)
|
|
59
69
|
raise FloatDomainError, "Computation results to 'NaN'"
|
|
60
70
|
end
|
|
61
71
|
BigDecimal::NAN
|
|
62
72
|
end
|
|
73
|
+
|
|
74
|
+
# Iteration for Newton's method with increasing precision
|
|
75
|
+
def self.newton_loop(prec, initial_precision: BigDecimal.double_fig / 2, safe_margin: 2) # :nodoc:
|
|
76
|
+
precs = []
|
|
77
|
+
while prec > initial_precision
|
|
78
|
+
precs << prec
|
|
79
|
+
prec = (precs.last + 1) / 2 + safe_margin
|
|
80
|
+
end
|
|
81
|
+
precs.reverse_each do |p|
|
|
82
|
+
yield p
|
|
83
|
+
end
|
|
84
|
+
end
|
|
85
|
+
|
|
86
|
+
# Calculates Math.log(x.to_f) considering large or small exponent
|
|
87
|
+
def self.float_log(x) # :nodoc:
|
|
88
|
+
Math.log(x._decimal_shift(-x.exponent).to_f) + x.exponent * Math.log(10)
|
|
89
|
+
end
|
|
90
|
+
|
|
91
|
+
# Calculating Taylor series sum using binary splitting method
|
|
92
|
+
# Calculates f(x) = (x/d0)*(1+(x/d1)*(1+(x/d2)*(1+(x/d3)*(1+...))))
|
|
93
|
+
# x.n_significant_digits or ds.size must be small to be performant.
|
|
94
|
+
def self.taylor_sum_binary_splitting(x, ds, prec) # :nodoc:
|
|
95
|
+
fs = ds.map {|d| [0, BigDecimal(d)] }
|
|
96
|
+
# fs = [[a0, a1], [b0, b1], [c0, c1], ...]
|
|
97
|
+
# f(x) = a0/a1+(x/a1)*(1+b0/b1+(x/b1)*(1+c0/c1+(x/c1)*(1+d0/d1+(x/d1)*(1+...))))
|
|
98
|
+
while fs.size > 1
|
|
99
|
+
# Merge two adjacent fractions
|
|
100
|
+
# from: (1 + a0/a1 + x/a1 * (1 + b0/b1 + x/b1 * rest))
|
|
101
|
+
# to: (1 + (a0*b1+x*(b0+b1))/(a1*b1) + (x*x)/(a1*b1) * rest)
|
|
102
|
+
xn = xn ? xn.mult(xn, prec) : x
|
|
103
|
+
fs = fs.each_slice(2).map do |(a, b)|
|
|
104
|
+
b ||= [0, BigDecimal(1)._decimal_shift([xn.exponent, 0].max + 2)]
|
|
105
|
+
[
|
|
106
|
+
(a[0] * b[1]).add(xn * (b[0] + b[1]), prec),
|
|
107
|
+
a[1].mult(b[1], prec)
|
|
108
|
+
]
|
|
109
|
+
end
|
|
110
|
+
end
|
|
111
|
+
BigDecimal(fs[0][0]).div(fs[0][1], prec)
|
|
112
|
+
end
|
|
63
113
|
end
|
|
64
114
|
|
|
65
115
|
# call-seq:
|
|
@@ -170,7 +220,7 @@ class BigDecimal
|
|
|
170
220
|
result_prec = prec.nonzero? || [x.n_significant_digits, y.n_significant_digits, BigDecimal.double_fig].max + BigDecimal.double_fig
|
|
171
221
|
result_prec = [result_prec, limit].min if prec.zero? && limit.nonzero?
|
|
172
222
|
|
|
173
|
-
prec2 = result_prec + BigDecimal
|
|
223
|
+
prec2 = result_prec + BigDecimal::Internal::EXTRA_PREC
|
|
174
224
|
|
|
175
225
|
if y < 0
|
|
176
226
|
inv = x.power(-y, prec2)
|
|
@@ -226,9 +276,7 @@ class BigDecimal
|
|
|
226
276
|
ex = exponent / 2
|
|
227
277
|
x = _decimal_shift(-2 * ex)
|
|
228
278
|
y = BigDecimal(Math.sqrt(x.to_f), 0)
|
|
229
|
-
|
|
230
|
-
precs << 2 + precs.last / 2 while precs.last > BigDecimal.double_fig
|
|
231
|
-
precs.reverse_each do |p|
|
|
279
|
+
Internal.newton_loop(prec + BigDecimal::Internal::EXTRA_PREC) do |p|
|
|
232
280
|
y = y.add(x.div(y, p), p).div(2, p)
|
|
233
281
|
end
|
|
234
282
|
y._decimal_shift(ex).mult(1, prec)
|
|
@@ -263,60 +311,36 @@ module BigMath
|
|
|
263
311
|
return BigDecimal::Internal.infinity_computation_result if x.infinite?
|
|
264
312
|
return BigDecimal(0) if x == 1
|
|
265
313
|
|
|
266
|
-
prec2 = prec + BigDecimal
|
|
267
|
-
BigDecimal.save_limit do
|
|
268
|
-
BigDecimal.limit(0)
|
|
269
|
-
if x > 10 || x < 0.1
|
|
270
|
-
log10 = log(BigDecimal(10), prec2)
|
|
271
|
-
exponent = x.exponent
|
|
272
|
-
x = x._decimal_shift(-exponent)
|
|
273
|
-
if x < 0.3
|
|
274
|
-
x *= 10
|
|
275
|
-
exponent -= 1
|
|
276
|
-
end
|
|
277
|
-
return (log10 * exponent).add(log(x, prec2), prec)
|
|
278
|
-
end
|
|
279
|
-
|
|
280
|
-
x_minus_one_exponent = (x - 1).exponent
|
|
281
|
-
|
|
282
|
-
# log(x) = log(sqrt(sqrt(sqrt(sqrt(x))))) * 2**sqrt_steps
|
|
283
|
-
sqrt_steps = [Integer.sqrt(prec2) + 3 * x_minus_one_exponent, 0].max
|
|
314
|
+
prec2 = prec + BigDecimal::Internal::EXTRA_PREC
|
|
284
315
|
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
# log(x) = -log((1 + X) / (1 - X)) where X = (x - 1) / (x + 1)
|
|
294
|
-
# log(x) = 2 * (X + X**3 / 3 + X**5 / 5 + X**7 / 7 + ...)
|
|
295
|
-
x = (x - 1).div(x + 1, sqrt_prec)
|
|
296
|
-
y = x
|
|
297
|
-
x2 = x.mult(x, prec2)
|
|
298
|
-
1.step do |i|
|
|
299
|
-
n = prec2 + x.exponent - y.exponent + x2.exponent
|
|
300
|
-
break if n <= 0 || x.zero?
|
|
301
|
-
x = x.mult(x2.round(n - x2.exponent), n)
|
|
302
|
-
y = y.add(x.div(2 * i + 1, n), prec2)
|
|
303
|
-
end
|
|
316
|
+
# Reduce x to near 1
|
|
317
|
+
if x > 1.01 || x < 0.99
|
|
318
|
+
# log(x) = log(x/exp(logx_approx)) + logx_approx
|
|
319
|
+
logx_approx = BigDecimal(BigDecimal::Internal.float_log(x), 0)
|
|
320
|
+
x = x.div(exp(logx_approx, prec2), prec2)
|
|
321
|
+
else
|
|
322
|
+
logx_approx = BigDecimal(0)
|
|
323
|
+
end
|
|
304
324
|
|
|
305
|
-
|
|
325
|
+
# Solve exp(y) - x = 0 with Newton's method
|
|
326
|
+
# Repeat: y -= (exp(y) - x) / exp(y)
|
|
327
|
+
y = BigDecimal(BigDecimal::Internal.float_log(x), 0)
|
|
328
|
+
exp_additional_prec = [-(x - 1).exponent, 0].max
|
|
329
|
+
BigDecimal::Internal.newton_loop(prec2) do |p|
|
|
330
|
+
expy = exp(y, p + exp_additional_prec)
|
|
331
|
+
y = y.sub(expy.sub(x, p).div(expy, p), p)
|
|
306
332
|
end
|
|
333
|
+
y.add(logx_approx, prec)
|
|
307
334
|
end
|
|
308
335
|
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
y = y.add(xn, prec)
|
|
318
|
-
end
|
|
319
|
-
y
|
|
336
|
+
private_class_method def _exp_binary_splitting(x, prec) # :nodoc:
|
|
337
|
+
return BigDecimal(1) if x.zero?
|
|
338
|
+
# Find k that satisfies x**k / k! < 10**(-prec)
|
|
339
|
+
log10 = Math.log(10)
|
|
340
|
+
logx = BigDecimal::Internal.float_log(x.abs)
|
|
341
|
+
step = (1..).bsearch { |k| Math.lgamma(k + 1)[0] - k * logx > prec * log10 }
|
|
342
|
+
# exp(x)-1 = x*(1+x/2*(1+x/3*(1+x/4*(1+x/5*(1+...)))))
|
|
343
|
+
1 + BigDecimal::Internal.taylor_sum_binary_splitting(x, [*1..step], prec)
|
|
320
344
|
end
|
|
321
345
|
|
|
322
346
|
# call-seq:
|
|
@@ -333,19 +357,39 @@ module BigMath
|
|
|
333
357
|
prec = BigDecimal::Internal.coerce_validate_prec(prec, :exp)
|
|
334
358
|
x = BigDecimal::Internal.coerce_to_bigdecimal(x, prec, :exp)
|
|
335
359
|
return BigDecimal::Internal.nan_computation_result if x.nan?
|
|
336
|
-
|
|
360
|
+
if x.infinite? || x.exponent >= 21 # exp(10**20) and exp(-10**20) overflows/underflows 64-bit exponent
|
|
361
|
+
if x.positive?
|
|
362
|
+
return BigDecimal::Internal.infinity_computation_result
|
|
363
|
+
elsif x.infinite?
|
|
364
|
+
# exp(-Infinity) is +0 by definition, this is not an underflow.
|
|
365
|
+
return BigDecimal(0)
|
|
366
|
+
else
|
|
367
|
+
return BigDecimal::Internal.underflow_computation_result
|
|
368
|
+
end
|
|
369
|
+
end
|
|
370
|
+
|
|
337
371
|
return BigDecimal(1) if x.zero?
|
|
338
372
|
|
|
339
373
|
# exp(x * 10**cnt) = exp(x)**(10**cnt)
|
|
340
374
|
cnt = x < -1 || x > 1 ? x.exponent : 0
|
|
341
|
-
prec2 = prec + BigDecimal
|
|
375
|
+
prec2 = prec + BigDecimal::Internal::EXTRA_PREC + cnt
|
|
342
376
|
x = x._decimal_shift(-cnt)
|
|
343
377
|
|
|
344
|
-
#
|
|
345
|
-
#
|
|
346
|
-
# exp(x)
|
|
347
|
-
|
|
348
|
-
|
|
378
|
+
# Decimal form of bit-burst algorithm
|
|
379
|
+
# Calculate exp(x.xxxxxxxxxxxxxxxx) as
|
|
380
|
+
# exp(x.xx) * exp(0.00xx) * exp(0.0000xxxx) * exp(0.00000000xxxxxxxx)
|
|
381
|
+
x = x.mult(1, prec2)
|
|
382
|
+
n = 2
|
|
383
|
+
y = BigDecimal(1)
|
|
384
|
+
BigDecimal.save_limit do
|
|
385
|
+
BigDecimal.limit(0)
|
|
386
|
+
while x != 0 do
|
|
387
|
+
partial_x = x.truncate(n)
|
|
388
|
+
x -= partial_x
|
|
389
|
+
y = y.mult(_exp_binary_splitting(partial_x, prec2), prec2)
|
|
390
|
+
n *= 2
|
|
391
|
+
end
|
|
392
|
+
end
|
|
349
393
|
|
|
350
394
|
# calculate exp(x * 10**cnt) from exp(x)
|
|
351
395
|
# exp(x * 10**k) = exp(x * 10**(k - 1)) ** 10
|
data/sample/linear.rb
CHANGED
|
@@ -1,6 +1,3 @@
|
|
|
1
|
-
#!/usr/local/bin/ruby
|
|
2
|
-
# frozen_string_literal: false
|
|
3
|
-
|
|
4
1
|
#
|
|
5
2
|
# linear.rb
|
|
6
3
|
#
|
|
@@ -13,62 +10,101 @@
|
|
|
13
10
|
|
|
14
11
|
# :stopdoc:
|
|
15
12
|
require "bigdecimal"
|
|
16
|
-
require "bigdecimal/ludcmp"
|
|
17
13
|
|
|
18
|
-
#
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
#
|
|
14
|
+
# Requires gem matrix
|
|
15
|
+
require "matrix"
|
|
16
|
+
|
|
17
|
+
class PrecisionSpecifiedValue
|
|
18
|
+
# NOTE:
|
|
19
|
+
# Change following PREC if needed.
|
|
20
|
+
|
|
21
|
+
attr_reader :value
|
|
22
|
+
def initialize(value, prec)
|
|
23
|
+
@value = BigDecimal(value)
|
|
24
|
+
@prec = prec
|
|
25
|
+
end
|
|
26
|
+
|
|
27
|
+
def unwrap(value)
|
|
28
|
+
PrecisionSpecifiedValue === value ? value.value : value
|
|
29
|
+
end
|
|
30
|
+
|
|
31
|
+
def coerce(other)
|
|
32
|
+
[self.class.new(unwrap(other), @prec), self]
|
|
33
|
+
end
|
|
34
|
+
|
|
35
|
+
def abs
|
|
36
|
+
self.class.new(@value.abs, @prec)
|
|
37
|
+
end
|
|
38
|
+
|
|
39
|
+
def >(other)
|
|
40
|
+
@value > unwrap(other)
|
|
41
|
+
end
|
|
42
|
+
|
|
43
|
+
def <(other)
|
|
44
|
+
@value < unwrap(other)
|
|
45
|
+
end
|
|
46
|
+
|
|
47
|
+
def -(other)
|
|
48
|
+
self.class.new(@value.sub(unwrap(other), @prec), @prec)
|
|
49
|
+
end
|
|
50
|
+
|
|
51
|
+
def +(other)
|
|
52
|
+
self.class.new(@value.add(unwrap(other), @prec), @prec)
|
|
53
|
+
end
|
|
54
|
+
|
|
55
|
+
def *(other)
|
|
56
|
+
self.class.new(@value.mult(unwrap(other), @prec), @prec)
|
|
57
|
+
end
|
|
58
|
+
|
|
59
|
+
def quo(other)
|
|
60
|
+
self.class.new(@value.div(unwrap(other), @prec), @prec)
|
|
61
|
+
end
|
|
62
|
+
end
|
|
63
|
+
|
|
64
|
+
return if __FILE__ != $0
|
|
23
65
|
|
|
24
|
-
include LUSolve
|
|
25
66
|
def rd_order(na)
|
|
26
|
-
|
|
27
|
-
|
|
67
|
+
printf("Number of equations ?") if(na <= 0)
|
|
68
|
+
ARGF.gets().to_i
|
|
28
69
|
end
|
|
29
70
|
|
|
30
|
-
na
|
|
31
|
-
zero = BigDecimal("0.0")
|
|
32
|
-
one = BigDecimal("1.0")
|
|
71
|
+
na = ARGV.size
|
|
33
72
|
|
|
34
73
|
while (n=rd_order(na))>0
|
|
35
74
|
a = []
|
|
36
|
-
as= []
|
|
37
75
|
b = []
|
|
38
76
|
if na <= 0
|
|
39
77
|
# Read data from console.
|
|
40
78
|
printf("\nEnter coefficient matrix element A[i,j]\n")
|
|
41
79
|
for i in 0...n do
|
|
42
|
-
|
|
80
|
+
a << n.times.map do |j|
|
|
43
81
|
printf("A[%d,%d]? ",i,j); s = ARGF.gets
|
|
44
|
-
|
|
45
|
-
as << BigDecimal(s)
|
|
82
|
+
BigDecimal(s)
|
|
46
83
|
end
|
|
47
84
|
printf("Contatant vector element b[%d] ? ",i)
|
|
48
85
|
b << BigDecimal(ARGF.gets)
|
|
49
86
|
end
|
|
50
87
|
else
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
end
|
|
61
|
-
b << BigDecimal(s[n])
|
|
62
|
-
end
|
|
88
|
+
# Read data from specified file.
|
|
89
|
+
printf("Coefficient matrix and constant vector.\n")
|
|
90
|
+
for i in 0...n do
|
|
91
|
+
s = ARGF.gets
|
|
92
|
+
printf("%d) %s",i,s)
|
|
93
|
+
s = s.split
|
|
94
|
+
a << n.times.map {|j| BigDecimal(s[j]) }
|
|
95
|
+
b << BigDecimal(s[n])
|
|
96
|
+
end
|
|
63
97
|
end
|
|
64
|
-
|
|
98
|
+
|
|
99
|
+
prec = 100
|
|
100
|
+
matrix = Matrix[*a.map {|row| row.map {|v| PrecisionSpecifiedValue.new(v, prec) } }]
|
|
101
|
+
vector = b.map {|v| PrecisionSpecifiedValue.new(v, prec) }
|
|
102
|
+
x = matrix.lup.solve(vector).map(&:value)
|
|
103
|
+
|
|
65
104
|
printf("Answer(x[i] & (A*x-b)[i]) follows\n")
|
|
66
105
|
for i in 0...n do
|
|
67
106
|
printf("x[%d]=%s ",i,x[i].to_s)
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
s = s + as[i*n+j]*x[j]
|
|
71
|
-
end
|
|
72
|
-
printf(" & %s\n",(s-b[i]).to_s)
|
|
107
|
+
diff = a[i].zip(x).sum {|aij, xj| aij*xj }.sub(b[i], 10)
|
|
108
|
+
printf(" & %s\n", diff.to_s)
|
|
73
109
|
end
|
|
74
110
|
end
|
data/sample/nlsolve.rb
CHANGED
|
@@ -1,40 +1,57 @@
|
|
|
1
|
-
#!/usr/local/bin/ruby
|
|
2
|
-
# frozen_string_literal: false
|
|
3
|
-
|
|
4
1
|
#
|
|
5
2
|
# nlsolve.rb
|
|
6
3
|
# An example for solving nonlinear algebraic equation system.
|
|
7
4
|
#
|
|
8
5
|
|
|
9
6
|
require "bigdecimal"
|
|
10
|
-
|
|
11
|
-
include Newton
|
|
7
|
+
require_relative "linear"
|
|
12
8
|
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
9
|
+
# Requires gem matrix
|
|
10
|
+
require "matrix"
|
|
11
|
+
|
|
12
|
+
# :stopdoc:
|
|
13
|
+
|
|
14
|
+
def func((x, y)) # defines functions solved
|
|
15
|
+
[
|
|
16
|
+
x**2 + y**2 - 2,
|
|
17
|
+
(x - 1)**2 + (y + 1)**2 - 3
|
|
18
|
+
]
|
|
19
|
+
end
|
|
20
|
+
|
|
21
|
+
def jacobian(x, f, delta, prec)
|
|
22
|
+
dim = x.size
|
|
23
|
+
dim.times.map do |i|
|
|
24
|
+
xplus = Array.new(dim) {|j| x[i] + (j == i ? delta : 0) }
|
|
25
|
+
xminus = Array.new(dim) {|j| x[i] - (j == i ? delta : 0) }
|
|
26
|
+
yplus = f.call(xplus)
|
|
27
|
+
yminus = f.call(xminus)
|
|
28
|
+
yplus.zip(yminus).map {|p, m| (p - m).div(2 * delta, prec) }
|
|
29
|
+
end.transpose
|
|
30
|
+
end
|
|
31
|
+
|
|
32
|
+
def nlsolve(initial_x, prec:, max_iteration: 100, &f)
|
|
33
|
+
initial_x = initial_x.map {|v| BigDecimal(v) }
|
|
34
|
+
x = initial_x
|
|
35
|
+
delta = BigDecimal(0.01)
|
|
36
|
+
calc_prec = prec + 10
|
|
37
|
+
max_iteration.times do |iteration|
|
|
38
|
+
# Newton step
|
|
39
|
+
jacobian = jacobian(x, f, delta, calc_prec)
|
|
40
|
+
matrix = Matrix[*jacobian.map {|row| row.map {|v| PrecisionSpecifiedValue.new(v, calc_prec) } }]
|
|
41
|
+
y = f.call(x)
|
|
42
|
+
vector = y.map {|v| PrecisionSpecifiedValue.new(v, calc_prec) }
|
|
43
|
+
dx = matrix.lup.solve(vector).map(&:value)
|
|
44
|
+
x_prev = x
|
|
45
|
+
x = x.zip(dx).map {|xi, di| xi.sub(di, prec) }
|
|
46
|
+
movement = x_prev.zip(x).map {|xn, xi| (xn - xi).abs }.max
|
|
47
|
+
delta = [movement, delta].min.mult(1, 10)
|
|
48
|
+
break if movement.zero? || movement.exponent < -prec
|
|
33
49
|
end
|
|
50
|
+
x
|
|
34
51
|
end
|
|
35
52
|
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
p
|
|
53
|
+
initial_value = [1, 1]
|
|
54
|
+
ans = nlsolve(initial_value, prec: 100) {|x| func(x) }
|
|
55
|
+
diff = func(ans).map {|v| v.mult(1, 10) }
|
|
56
|
+
p(ans:)
|
|
57
|
+
p(diff:)
|
data/sample/pi.rb
CHANGED
|
@@ -1,21 +1,16 @@
|
|
|
1
|
-
#!/usr/local/bin/ruby
|
|
2
|
-
# frozen_string_literal: false
|
|
3
|
-
|
|
4
1
|
#
|
|
5
2
|
# pi.rb
|
|
6
3
|
#
|
|
7
4
|
# Calculates 3.1415.... (the number of times that a circle's diameter
|
|
8
|
-
# will fit around the circle)
|
|
5
|
+
# will fit around the circle)
|
|
9
6
|
#
|
|
10
7
|
|
|
11
8
|
require "bigdecimal"
|
|
12
9
|
require "bigdecimal/math.rb"
|
|
13
10
|
|
|
14
|
-
include BigMath
|
|
15
|
-
|
|
16
11
|
if ARGV.size == 1
|
|
17
12
|
print "PI("+ARGV[0]+"):\n"
|
|
18
|
-
p PI(ARGV[0].to_i)
|
|
13
|
+
p BigMath.PI(ARGV[0].to_i)
|
|
19
14
|
else
|
|
20
15
|
print "TRY: ruby pi.rb 1000 \n"
|
|
21
16
|
end
|