bigdecimal 4.0.1-java → 4.1.0-java

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.
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.
@@ -60,6 +63,46 @@ class BigDecimal
60
63
  end
61
64
  BigDecimal::NAN
62
65
  end
66
+
67
+ # Iteration for Newton's method with increasing precision
68
+ def self.newton_loop(prec, initial_precision: BigDecimal.double_fig / 2, safe_margin: 2) # :nodoc:
69
+ precs = []
70
+ while prec > initial_precision
71
+ precs << prec
72
+ prec = (precs.last + 1) / 2 + safe_margin
73
+ end
74
+ precs.reverse_each do |p|
75
+ yield p
76
+ end
77
+ end
78
+
79
+ # Calculates Math.log(x.to_f) considering large or small exponent
80
+ def self.float_log(x) # :nodoc:
81
+ Math.log(x._decimal_shift(-x.exponent).to_f) + x.exponent * Math.log(10)
82
+ end
83
+
84
+ # Calculating Taylor series sum using binary splitting method
85
+ # Calculates f(x) = (x/d0)*(1+(x/d1)*(1+(x/d2)*(1+(x/d3)*(1+...))))
86
+ # x.n_significant_digits or ds.size must be small to be performant.
87
+ def self.taylor_sum_binary_splitting(x, ds, prec) # :nodoc:
88
+ fs = ds.map {|d| [0, BigDecimal(d)] }
89
+ # fs = [[a0, a1], [b0, b1], [c0, c1], ...]
90
+ # f(x) = a0/a1+(x/a1)*(1+b0/b1+(x/b1)*(1+c0/c1+(x/c1)*(1+d0/d1+(x/d1)*(1+...))))
91
+ while fs.size > 1
92
+ # Merge two adjacent fractions
93
+ # from: (1 + a0/a1 + x/a1 * (1 + b0/b1 + x/b1 * rest))
94
+ # to: (1 + (a0*b1+x*(b0+b1))/(a1*b1) + (x*x)/(a1*b1) * rest)
95
+ xn = xn ? xn.mult(xn, prec) : x
96
+ fs = fs.each_slice(2).map do |(a, b)|
97
+ b ||= [0, BigDecimal(1)._decimal_shift([xn.exponent, 0].max + 2)]
98
+ [
99
+ (a[0] * b[1]).add(xn * (b[0] + b[1]), prec),
100
+ a[1].mult(b[1], prec)
101
+ ]
102
+ end
103
+ end
104
+ BigDecimal(fs[0][0]).div(fs[0][1], prec)
105
+ end
63
106
  end
64
107
 
65
108
  # call-seq:
@@ -170,7 +213,7 @@ class BigDecimal
170
213
  result_prec = prec.nonzero? || [x.n_significant_digits, y.n_significant_digits, BigDecimal.double_fig].max + BigDecimal.double_fig
171
214
  result_prec = [result_prec, limit].min if prec.zero? && limit.nonzero?
172
215
 
173
- prec2 = result_prec + BigDecimal.double_fig
216
+ prec2 = result_prec + BigDecimal::Internal::EXTRA_PREC
174
217
 
175
218
  if y < 0
176
219
  inv = x.power(-y, prec2)
@@ -226,9 +269,7 @@ class BigDecimal
226
269
  ex = exponent / 2
227
270
  x = _decimal_shift(-2 * ex)
228
271
  y = BigDecimal(Math.sqrt(x.to_f), 0)
229
- precs = [prec + BigDecimal.double_fig]
230
- precs << 2 + precs.last / 2 while precs.last > BigDecimal.double_fig
231
- precs.reverse_each do |p|
272
+ Internal.newton_loop(prec + BigDecimal::Internal::EXTRA_PREC) do |p|
232
273
  y = y.add(x.div(y, p), p).div(2, p)
233
274
  end
234
275
  y._decimal_shift(ex).mult(1, prec)
@@ -263,60 +304,36 @@ module BigMath
263
304
  return BigDecimal::Internal.infinity_computation_result if x.infinite?
264
305
  return BigDecimal(0) if x == 1
265
306
 
266
- prec2 = prec + BigDecimal.double_fig
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
307
+ prec2 = prec + BigDecimal::Internal::EXTRA_PREC
281
308
 
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
284
-
285
- lg2 = 0.3010299956639812
286
- sqrt_prec = prec2 + [-x_minus_one_exponent, 0].max + (sqrt_steps * lg2).ceil
287
-
288
- sqrt_steps.times do
289
- x = x.sqrt(sqrt_prec)
290
- end
291
-
292
- # Taylor series for log(x) around 1
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
309
+ # Reduce x to near 1
310
+ if x > 1.01 || x < 0.99
311
+ # log(x) = log(x/exp(logx_approx)) + logx_approx
312
+ logx_approx = BigDecimal(BigDecimal::Internal.float_log(x), 0)
313
+ x = x.div(exp(logx_approx, prec2), prec2)
314
+ else
315
+ logx_approx = BigDecimal(0)
316
+ end
304
317
 
305
- y.mult(2 ** (sqrt_steps + 1), prec)
318
+ # Solve exp(y) - x = 0 with Newton's method
319
+ # Repeat: y -= (exp(y) - x) / exp(y)
320
+ y = BigDecimal(BigDecimal::Internal.float_log(x), 0)
321
+ exp_additional_prec = [-(x - 1).exponent, 0].max
322
+ BigDecimal::Internal.newton_loop(prec2) do |p|
323
+ expy = exp(y, p + exp_additional_prec)
324
+ y = y.sub(expy.sub(x, p).div(expy, p), p)
306
325
  end
326
+ y.add(logx_approx, prec)
307
327
  end
308
328
 
309
- # Taylor series for exp(x) around 0
310
- private_class_method def _exp_taylor(x, prec) # :nodoc:
311
- xn = BigDecimal(1)
312
- y = BigDecimal(1)
313
- 1.step do |i|
314
- n = prec + xn.exponent
315
- break if n <= 0 || xn.zero?
316
- xn = xn.mult(x, n).div(i, n)
317
- y = y.add(xn, prec)
318
- end
319
- y
329
+ private_class_method def _exp_binary_splitting(x, prec) # :nodoc:
330
+ return BigDecimal(1) if x.zero?
331
+ # Find k that satisfies x**k / k! < 10**(-prec)
332
+ log10 = Math.log(10)
333
+ logx = BigDecimal::Internal.float_log(x.abs)
334
+ step = (1..).bsearch { |k| Math.lgamma(k + 1)[0] - k * logx > prec * log10 }
335
+ # exp(x)-1 = x*(1+x/2*(1+x/3*(1+x/4*(1+x/5*(1+...)))))
336
+ 1 + BigDecimal::Internal.taylor_sum_binary_splitting(x, [*1..step], prec)
320
337
  end
321
338
 
322
339
  # call-seq:
@@ -338,14 +355,24 @@ module BigMath
338
355
 
339
356
  # exp(x * 10**cnt) = exp(x)**(10**cnt)
340
357
  cnt = x < -1 || x > 1 ? x.exponent : 0
341
- prec2 = prec + BigDecimal.double_fig + cnt
358
+ prec2 = prec + BigDecimal::Internal::EXTRA_PREC + cnt
342
359
  x = x._decimal_shift(-cnt)
343
360
 
344
- # Calculation of exp(small_prec) is fast because calculation of x**n is fast
345
- # Calculation of exp(small_abs) converges fast.
346
- # exp(x) = exp(small_prec_part + small_abs_part) = exp(small_prec_part) * exp(small_abs_part)
347
- x_small_prec = x.round(Integer.sqrt(prec2))
348
- y = _exp_taylor(x_small_prec, prec2).mult(_exp_taylor(x.sub(x_small_prec, prec2), prec2), prec2)
361
+ # Decimal form of bit-burst algorithm
362
+ # Calculate exp(x.xxxxxxxxxxxxxxxx) as
363
+ # exp(x.xx) * exp(0.00xx) * exp(0.0000xxxx) * exp(0.00000000xxxxxxxx)
364
+ x = x.mult(1, prec2)
365
+ n = 2
366
+ y = BigDecimal(1)
367
+ BigDecimal.save_limit do
368
+ BigDecimal.limit(0)
369
+ while x != 0 do
370
+ partial_x = x.truncate(n)
371
+ x -= partial_x
372
+ y = y.mult(_exp_binary_splitting(partial_x, prec2), prec2)
373
+ n *= 2
374
+ end
375
+ end
349
376
 
350
377
  # calculate exp(x * 10**cnt) from exp(x)
351
378
  # 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
- # NOTE:
20
- # Change following BigDecimal.limit() if needed.
21
- BigDecimal.limit(100)
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
- printf("Number of equations ?") if(na <= 0)
27
- n = ARGF.gets().to_i
67
+ printf("Number of equations ?") if(na <= 0)
68
+ ARGF.gets().to_i
28
69
  end
29
70
 
30
- na = ARGV.size
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
- for j in 0...n do
80
+ a << n.times.map do |j|
43
81
  printf("A[%d,%d]? ",i,j); s = ARGF.gets
44
- a << BigDecimal(s)
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
- # Read data from specified file.
52
- printf("Coefficient matrix and constant vector.\n")
53
- for i in 0...n do
54
- s = ARGF.gets
55
- printf("%d) %s",i,s)
56
- s = s.split
57
- for j in 0...n do
58
- a << BigDecimal(s[j])
59
- as << BigDecimal(s[j])
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
- x = lusolve(a,b,ludecomp(a,n,zero,one),zero)
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
- s = zero
69
- for j in 0...n do
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
- require "bigdecimal/newton"
11
- include Newton
7
+ require_relative "linear"
12
8
 
13
- class Function # :nodoc: all
14
- def initialize()
15
- @zero = BigDecimal("0.0")
16
- @one = BigDecimal("1.0")
17
- @two = BigDecimal("2.0")
18
- @ten = BigDecimal("10.0")
19
- @eps = BigDecimal("1.0e-16")
20
- end
21
- def zero;@zero;end
22
- def one ;@one ;end
23
- def two ;@two ;end
24
- def ten ;@ten ;end
25
- def eps ;@eps ;end
26
- def values(x) # <= defines functions solved
27
- f = []
28
- f1 = x[0]*x[0] + x[1]*x[1] - @two # f1 = x**2 + y**2 - 2 => 0
29
- f2 = x[0] - x[1] # f2 = x - y => 0
30
- f <<= f1
31
- f <<= f2
32
- f
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
- f = BigDecimal.limit(100)
37
- f = Function.new
38
- x = [f.zero,f.zero] # Initial values
39
- n = nlsolve(f,x)
40
- p x
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) using J. Machin's formula.
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