bigdecimal 3.3.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.
@@ -2,6 +2,8 @@
2
2
  require "bigdecimal/ludcmp"
3
3
  require "bigdecimal/jacobian"
4
4
 
5
+ warn "'bigdecimal/newton' is deprecated and will be removed in a future release."
6
+
5
7
  #
6
8
  # newton.rb
7
9
  #
@@ -1,4 +1,4 @@
1
- # frozen_string_literal: false
1
+ # frozen_string_literal: true
2
2
  #
3
3
  #--
4
4
  # bigdecimal/util extends various native classes to provide the #to_d method,
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:
@@ -144,10 +194,10 @@ class BigDecimal
144
194
  return BigDecimal(1)
145
195
  end
146
196
 
147
- prec = BigDecimal.limit if prec.zero?
197
+ limit = BigDecimal.limit
148
198
  frac_part = y.frac
149
199
 
150
- if frac_part.zero? && prec.zero?
200
+ if frac_part.zero? && prec.zero? && limit.zero?
151
201
  # Infinite precision calculation for `x ** int` and `x.power(int)`
152
202
  int_part = y.fix.to_i
153
203
  int_part = -int_part if (neg = int_part < 0)
@@ -167,18 +217,19 @@ class BigDecimal
167
217
  return neg ? BigDecimal(1) / ans : ans
168
218
  end
169
219
 
170
- prec = [x.n_significant_digits, y.n_significant_digits, BigDecimal.double_fig].max + BigDecimal.double_fig if prec.zero?
220
+ result_prec = prec.nonzero? || [x.n_significant_digits, y.n_significant_digits, BigDecimal.double_fig].max + BigDecimal.double_fig
221
+ result_prec = [result_prec, limit].min if prec.zero? && limit.nonzero?
222
+
223
+ prec2 = result_prec + BigDecimal::Internal::EXTRA_PREC
171
224
 
172
225
  if y < 0
173
- inv = x.power(-y, prec)
226
+ inv = x.power(-y, prec2)
174
227
  return BigDecimal(0) if inv.infinite?
175
228
  return BigDecimal::Internal.infinity_computation_result if inv.zero?
176
- return BigDecimal(1).div(inv, prec)
229
+ return BigDecimal(1).div(inv, result_prec)
177
230
  end
178
231
 
179
- prec2 = prec + BigDecimal.double_fig
180
-
181
- if frac_part.zero? && y.exponent < Math.log(prec) * 5 + 20
232
+ if frac_part.zero? && y.exponent < Math.log(result_prec) * 5 + 20
182
233
  # Use exponentiation by squaring if y is an integer and not too large
183
234
  pow_prec = prec2 + y.exponent
184
235
  n = 1
@@ -191,16 +242,16 @@ class BigDecimal
191
242
  break if n > int_part
192
243
  xn = xn.mult(xn, pow_prec)
193
244
  end
194
- ans.mult(1, prec)
245
+ ans.mult(1, result_prec)
195
246
  else
196
- if x > 1
247
+ if x > 1 && x.finite?
197
248
  # To calculate exp(z, prec), z needs prec+max(z.exponent, 0) precision if z > 0.
198
249
  # Estimate (y*log(x)).exponent
199
250
  logx_exponent = x < 2 ? (x - 1).exponent : Math.log10(x.exponent).round
200
251
  ylogx_exponent = y.exponent + logx_exponent
201
252
  prec2 += [ylogx_exponent, 0].max
202
253
  end
203
- BigMath.exp(BigMath.log(x, prec2).mult(y, prec2), prec)
254
+ BigMath.exp(BigMath.log(x, prec2).mult(y, prec2), result_prec)
204
255
  end
205
256
  end
206
257
 
@@ -217,15 +268,15 @@ class BigDecimal
217
268
  return self if zero?
218
269
 
219
270
  if prec == 0
220
- prec = BigDecimal.limit.nonzero? || n_significant_digits + BigDecimal.double_fig
271
+ limit = BigDecimal.limit
272
+ prec = n_significant_digits + BigDecimal.double_fig
273
+ prec = [limit, prec].min if limit.nonzero?
221
274
  end
222
275
 
223
276
  ex = exponent / 2
224
277
  x = _decimal_shift(-2 * ex)
225
278
  y = BigDecimal(Math.sqrt(x.to_f), 0)
226
- precs = [prec + BigDecimal.double_fig]
227
- precs << 2 + precs.last / 2 while precs.last > BigDecimal.double_fig
228
- precs.reverse_each do |p|
279
+ Internal.newton_loop(prec + BigDecimal::Internal::EXTRA_PREC) do |p|
229
280
  y = y.add(x.div(y, p), p).div(2, p)
230
281
  end
231
282
  y._decimal_shift(ex).mult(1, prec)
@@ -235,6 +286,7 @@ end
235
286
  # Core BigMath methods for BigDecimal (log, exp) are defined here.
236
287
  # Other methods (sin, cos, atan) are defined in 'bigdecimal/math.rb'.
237
288
  module BigMath
289
+ module_function
238
290
 
239
291
  # call-seq:
240
292
  # BigMath.log(decimal, numeric) -> BigDecimal
@@ -248,7 +300,7 @@ module BigMath
248
300
  #
249
301
  # If +decimal+ is NaN, returns NaN.
250
302
  #
251
- def self.log(x, prec)
303
+ def log(x, prec)
252
304
  prec = BigDecimal::Internal.coerce_validate_prec(prec, :log)
253
305
  raise Math::DomainError, 'Complex argument for BigMath.log' if Complex === x
254
306
 
@@ -259,60 +311,36 @@ module BigMath
259
311
  return BigDecimal::Internal.infinity_computation_result if x.infinite?
260
312
  return BigDecimal(0) if x == 1
261
313
 
262
- prec2 = prec + BigDecimal.double_fig
263
- BigDecimal.save_limit do
264
- BigDecimal.limit(0)
265
- if x > 10 || x < 0.1
266
- log10 = log(BigDecimal(10), prec2)
267
- exponent = x.exponent
268
- x = x._decimal_shift(-exponent)
269
- if x < 0.3
270
- x *= 10
271
- exponent -= 1
272
- end
273
- return (log10 * exponent).add(log(x, prec2), prec)
274
- end
275
-
276
- x_minus_one_exponent = (x - 1).exponent
314
+ prec2 = prec + BigDecimal::Internal::EXTRA_PREC
277
315
 
278
- # log(x) = log(sqrt(sqrt(sqrt(sqrt(x))))) * 2**sqrt_steps
279
- sqrt_steps = [Integer.sqrt(prec2) + 3 * x_minus_one_exponent, 0].max
280
-
281
- lg2 = 0.3010299956639812
282
- sqrt_prec = prec2 + [-x_minus_one_exponent, 0].max + (sqrt_steps * lg2).ceil
283
-
284
- sqrt_steps.times do
285
- x = x.sqrt(sqrt_prec)
286
- end
287
-
288
- # Taylor series for log(x) around 1
289
- # log(x) = -log((1 + X) / (1 - X)) where X = (x - 1) / (x + 1)
290
- # log(x) = 2 * (X + X**3 / 3 + X**5 / 5 + X**7 / 7 + ...)
291
- x = (x - 1).div(x + 1, sqrt_prec)
292
- y = x
293
- x2 = x.mult(x, prec2)
294
- 1.step do |i|
295
- n = prec2 + x.exponent - y.exponent + x2.exponent
296
- break if n <= 0 || x.zero?
297
- x = x.mult(x2.round(n - x2.exponent), n)
298
- y = y.add(x.div(2 * i + 1, n), prec2)
299
- 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
300
324
 
301
- y.mult(2 ** (sqrt_steps + 1), prec)
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)
302
332
  end
333
+ y.add(logx_approx, prec)
303
334
  end
304
335
 
305
- # Taylor series for exp(x) around 0
306
- private_class_method def self._exp_taylor(x, prec) # :nodoc:
307
- xn = BigDecimal(1)
308
- y = BigDecimal(1)
309
- 1.step do |i|
310
- n = prec + xn.exponent
311
- break if n <= 0 || xn.zero?
312
- xn = xn.mult(x, n).div(i, n)
313
- y = y.add(xn, prec)
314
- end
315
- 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)
316
344
  end
317
345
 
318
346
  # call-seq:
@@ -325,23 +353,43 @@ module BigMath
325
353
  #
326
354
  # If +decimal+ is NaN, returns NaN.
327
355
  #
328
- def self.exp(x, prec)
356
+ def exp(x, prec)
329
357
  prec = BigDecimal::Internal.coerce_validate_prec(prec, :exp)
330
358
  x = BigDecimal::Internal.coerce_to_bigdecimal(x, prec, :exp)
331
359
  return BigDecimal::Internal.nan_computation_result if x.nan?
332
- return x.positive? ? BigDecimal::Internal.infinity_computation_result : BigDecimal(0) if x.infinite?
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
+
333
371
  return BigDecimal(1) if x.zero?
334
372
 
335
373
  # exp(x * 10**cnt) = exp(x)**(10**cnt)
336
374
  cnt = x < -1 || x > 1 ? x.exponent : 0
337
- prec2 = prec + BigDecimal.double_fig + cnt
375
+ prec2 = prec + BigDecimal::Internal::EXTRA_PREC + cnt
338
376
  x = x._decimal_shift(-cnt)
339
377
 
340
- # Calculation of exp(small_prec) is fast because calculation of x**n is fast
341
- # Calculation of exp(small_abs) converges fast.
342
- # exp(x) = exp(small_prec_part + small_abs_part) = exp(small_prec_part) * exp(small_abs_part)
343
- x_small_prec = x.round(Integer.sqrt(prec2))
344
- y = _exp_taylor(x_small_prec, prec2).mult(_exp_taylor(x.sub(x_small_prec, prec2), prec2), prec2)
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
345
393
 
346
394
  # calculate exp(x * 10**cnt) from exp(x)
347
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
- # 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