nio 0.2.0
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/History.txt +5 -0
- data/License.txt +20 -0
- data/Manifest.txt +34 -0
- data/README.txt +560 -0
- data/Rakefile +4 -0
- data/SOURCE.txt +31 -0
- data/config/hoe.rb +73 -0
- data/config/requirements.rb +17 -0
- data/lib/nio/flttol.rb +654 -0
- data/lib/nio/fmt.rb +1872 -0
- data/lib/nio/repdec.rb +496 -0
- data/lib/nio/rtnlzr.rb +406 -0
- data/lib/nio/sugar.rb +99 -0
- data/lib/nio/tools.rb +44 -0
- data/lib/nio/version.rb +9 -0
- data/lib/nio.rb +8 -0
- data/log/debug.log +0 -0
- data/script/destroy +14 -0
- data/script/destroy.cmd +1 -0
- data/script/generate +14 -0
- data/script/generate.cmd +1 -0
- data/script/txt2html +74 -0
- data/script/txt2html.cmd +1 -0
- data/setup.rb +1585 -0
- data/tasks/deployment.rake +27 -0
- data/tasks/environment.rake +7 -0
- data/tasks/nuweb.rake +69 -0
- data/tasks/website.rake +17 -0
- data/test/data.yaml +101 -0
- data/test/test_fmt.rb +373 -0
- data/test/test_helper.rb +32 -0
- data/test/test_repdec.rb +88 -0
- data/test/test_rtnlzr.rb +125 -0
- data/test/test_tools.rb +40 -0
- metadata +88 -0
data/lib/nio/fmt.rb
ADDED
@@ -0,0 +1,1872 @@
|
|
1
|
+
# Formatting numbers as text
|
2
|
+
|
3
|
+
# Copyright (C) 2003-2005, Javier Goizueta <javier@goizueta.info>
|
4
|
+
#
|
5
|
+
# This program is free software; you can redistribute it and/or
|
6
|
+
# modify it under the terms of the GNU General Public License
|
7
|
+
# as published by the Free Software Foundation; either version 2
|
8
|
+
# of the License, or (at your option) any later version.
|
9
|
+
|
10
|
+
|
11
|
+
require 'nio/tools'
|
12
|
+
|
13
|
+
require 'nio/repdec'
|
14
|
+
|
15
|
+
require 'nio/rtnlzr'
|
16
|
+
|
17
|
+
require 'rational'
|
18
|
+
|
19
|
+
require 'bigdecimal'
|
20
|
+
|
21
|
+
module Nio
|
22
|
+
|
23
|
+
# positional notation, unformatted numeric literal: used as intermediate form
|
24
|
+
class NeutralNum
|
25
|
+
include StateEquivalent
|
26
|
+
def initialize(s='',d='',p=nil,r=nil,dgs=DigitsDef.base(10), inexact=false, round=:inf)
|
27
|
+
set s,d,p,r,dgs,dgs, inexact, round
|
28
|
+
end
|
29
|
+
attr_reader :sign, :digits, :dec_pos, :rep_pos, :special, :inexact, :rounding
|
30
|
+
attr_writer :sign, :digits, :dec_pos, :rep_pos, :special, :inexact, :rounding
|
31
|
+
|
32
|
+
# set number
|
33
|
+
def set(s,d,p=nil,r=nil,dgs=DigitsDef.base(10),inexact=false,rounding=:inf,normalize=true)
|
34
|
+
@sign = s # sign: '+','-',''
|
35
|
+
@digits = d # digits string
|
36
|
+
@dec_pos = p==nil ? d.length : p # position of decimal point: 0=before first digit...
|
37
|
+
@rep_pos = r==nil ? d.length : r # first repeated digit (0=first digit...)
|
38
|
+
@dgs = dgs
|
39
|
+
@base = @dgs.radix
|
40
|
+
@inexact = inexact
|
41
|
+
@special = nil
|
42
|
+
@rounding = rounding
|
43
|
+
trimZeros unless inexact
|
44
|
+
self
|
45
|
+
end
|
46
|
+
# set infinite (:inf) and invalid (:nan) numbers
|
47
|
+
def set_special(s,sgn='') # :inf, :nan
|
48
|
+
@special = s
|
49
|
+
@sign = sgn
|
50
|
+
self
|
51
|
+
end
|
52
|
+
|
53
|
+
def base
|
54
|
+
@base
|
55
|
+
end
|
56
|
+
def base_digits
|
57
|
+
@dgs
|
58
|
+
end
|
59
|
+
def base_digits=(dd)
|
60
|
+
@dgs = dd
|
61
|
+
@base = @dgs.radix
|
62
|
+
end
|
63
|
+
def base=(b)
|
64
|
+
@dgs = DigitsDef.base(b)
|
65
|
+
@base=@dgs.radix
|
66
|
+
end
|
67
|
+
|
68
|
+
# check for special numbers (which have only special and sign attributes)
|
69
|
+
def special?
|
70
|
+
special != nil
|
71
|
+
end
|
72
|
+
|
73
|
+
# check for special numbers (which have only special and sign attributes)
|
74
|
+
def inexact?
|
75
|
+
@inexact
|
76
|
+
end
|
77
|
+
|
78
|
+
def dup
|
79
|
+
n = NeutralNum.new
|
80
|
+
if special?
|
81
|
+
n.set_special @special.dup, @sign.dup
|
82
|
+
else
|
83
|
+
#n.set @sign.dup, @digits.dup, @dec_pos.dup, @rep_pos.dup, @dgs.dup
|
84
|
+
# in Ruby 1.6.8 Float,BigNum,Fixnum doesn't respond to dup
|
85
|
+
n.set @sign.dup, @digits.dup, @dec_pos, @rep_pos, @dgs.dup, @inexact, @rounding
|
86
|
+
end
|
87
|
+
return n
|
88
|
+
end
|
89
|
+
|
90
|
+
def zero?
|
91
|
+
z = false
|
92
|
+
if !special
|
93
|
+
if digits==''
|
94
|
+
z = true
|
95
|
+
else
|
96
|
+
z = true
|
97
|
+
for i in (0...@digits.length)
|
98
|
+
if dig_value(i)!=0
|
99
|
+
z = false
|
100
|
+
break
|
101
|
+
end
|
102
|
+
end
|
103
|
+
end
|
104
|
+
end
|
105
|
+
z
|
106
|
+
end
|
107
|
+
|
108
|
+
def round!(n, mode=:fix, dir=nil)
|
109
|
+
dir ||= rounding
|
110
|
+
trimLeadZeros
|
111
|
+
if n==:exact
|
112
|
+
return unless @inexact
|
113
|
+
n = @digits.size
|
114
|
+
end
|
115
|
+
|
116
|
+
n += @dec_pos if mode==:fix
|
117
|
+
n = [n,@digits.size].min if @inexact
|
118
|
+
|
119
|
+
adj = 0
|
120
|
+
dv = :tie
|
121
|
+
if @inexact && n==@digits.size
|
122
|
+
dv = @inexact==:roundup ? :hi : :lo
|
123
|
+
else
|
124
|
+
v = dig_value(n)
|
125
|
+
v2 = 2*v
|
126
|
+
if v2 < @base # v<((@base+1)/2)
|
127
|
+
dv = :lo
|
128
|
+
elsif v2 > @base # v>(@base/2)
|
129
|
+
dv = :hi
|
130
|
+
else
|
131
|
+
|
132
|
+
(n+1...@digits.length).each do |i|
|
133
|
+
if dig_value(i)>0
|
134
|
+
dv = :hi
|
135
|
+
break
|
136
|
+
end
|
137
|
+
end
|
138
|
+
|
139
|
+
dv = :hi if dv==:tie && @rep_pos<=n
|
140
|
+
end
|
141
|
+
end
|
142
|
+
|
143
|
+
if dv==:hi
|
144
|
+
adj = +1
|
145
|
+
elsif dv==:tie
|
146
|
+
if dir==:inf # towards nearest +/-infinity
|
147
|
+
adj = +1
|
148
|
+
elsif dir==:even # to nearest even digit (IEEE unbiased rounding)
|
149
|
+
adj = +1 if (dig_value(n-1)%2)!=0
|
150
|
+
elsif dir==:zero # towards zero
|
151
|
+
adj=0
|
152
|
+
# elsif dir==:odd
|
153
|
+
# adj = +1 unless (dig_value(n-1)%2)!=0
|
154
|
+
end
|
155
|
+
end
|
156
|
+
|
157
|
+
if n>@digits.length
|
158
|
+
(@digits.length...n).each do |i|
|
159
|
+
@digits << dig_char(dig_value(i))
|
160
|
+
@rep_pos += 1
|
161
|
+
end
|
162
|
+
end
|
163
|
+
|
164
|
+
prefix = ''
|
165
|
+
i = n-1
|
166
|
+
while adj!=0
|
167
|
+
v = dig_value(i)
|
168
|
+
v += adj
|
169
|
+
adj = 0
|
170
|
+
if v<0
|
171
|
+
v += @base
|
172
|
+
adj = -1
|
173
|
+
elsif v>=@base
|
174
|
+
v -= @base
|
175
|
+
adj = +1
|
176
|
+
end
|
177
|
+
if i<0
|
178
|
+
prefix = dig_char(v)+prefix
|
179
|
+
elsif i<@digits.length
|
180
|
+
@digits[i] = dig_char(v)
|
181
|
+
end
|
182
|
+
i += -1
|
183
|
+
end
|
184
|
+
|
185
|
+
if n<0
|
186
|
+
@digits = ""
|
187
|
+
else
|
188
|
+
@digits = @digits[0...n]
|
189
|
+
end
|
190
|
+
@rep_pos = @digits.length
|
191
|
+
|
192
|
+
if prefix!=''
|
193
|
+
@digits = prefix + @digits
|
194
|
+
@dec_pos += prefix.length
|
195
|
+
@rep_pos += prefix.length
|
196
|
+
end
|
197
|
+
|
198
|
+
|
199
|
+
end
|
200
|
+
|
201
|
+
def round(n, mode=:fix, dir=nil)
|
202
|
+
dir ||= rounding
|
203
|
+
nn = dup
|
204
|
+
nn.round!(n,mode,dir)
|
205
|
+
return nn
|
206
|
+
end
|
207
|
+
|
208
|
+
def trimTrailZeros()
|
209
|
+
i = @digits.length
|
210
|
+
while i>0 && dig_value(i-1)==0
|
211
|
+
i -= 1
|
212
|
+
end
|
213
|
+
if @rep_pos>=i
|
214
|
+
@digits = @digits[0...i]
|
215
|
+
@rep_pos = i
|
216
|
+
end
|
217
|
+
|
218
|
+
if @digits==''
|
219
|
+
@digits = dig_char(0) # '0'
|
220
|
+
@rep_pos = 1
|
221
|
+
@dec_pos = 1
|
222
|
+
end
|
223
|
+
|
224
|
+
end
|
225
|
+
|
226
|
+
def trimLeadZeros()
|
227
|
+
i = 0
|
228
|
+
while i<@digits.length && dig_value(i)==0
|
229
|
+
i += 1
|
230
|
+
end
|
231
|
+
@digits = @digits[i...@digits.length]
|
232
|
+
@dec_pos -= i
|
233
|
+
@rep_pos -= i
|
234
|
+
|
235
|
+
if @digits==''
|
236
|
+
@digits = dig_char(0) # '0'
|
237
|
+
@rep_pos = 1
|
238
|
+
@dec_pos = 1
|
239
|
+
end
|
240
|
+
|
241
|
+
end
|
242
|
+
|
243
|
+
def trimZeros()
|
244
|
+
trimLeadZeros
|
245
|
+
trimTrailZeros
|
246
|
+
end
|
247
|
+
|
248
|
+
protected
|
249
|
+
|
250
|
+
def dig_value(i)
|
251
|
+
v = 0
|
252
|
+
if i>=@rep_pos
|
253
|
+
i -= @digits.length
|
254
|
+
i %= @digits.length - @rep_pos if @rep_pos<@digits.length
|
255
|
+
i += @rep_pos
|
256
|
+
end
|
257
|
+
if i>=0 && i<@digits.length
|
258
|
+
v = @dgs.digit_value(@digits[i]) #digcode_value(@digits[i])
|
259
|
+
end
|
260
|
+
return v>=0 && v<@base ? v : nil
|
261
|
+
end
|
262
|
+
#def digcode_value(c)
|
263
|
+
# v = c-?0
|
264
|
+
# if v>9
|
265
|
+
# v = 10 + c.chr.downcase[0] - ?a
|
266
|
+
# end
|
267
|
+
# v
|
268
|
+
# @dgs.digit_value(c)
|
269
|
+
#end
|
270
|
+
|
271
|
+
def dig_char(v)
|
272
|
+
c = ''
|
273
|
+
if v!=nil && v>=0 && v<@base
|
274
|
+
c = @dgs.digit_char(v).chr
|
275
|
+
end
|
276
|
+
c
|
277
|
+
end
|
278
|
+
|
279
|
+
end
|
280
|
+
|
281
|
+
class NeutralNum
|
282
|
+
public
|
283
|
+
def to_RepDec
|
284
|
+
n = RepDec.new(@base)
|
285
|
+
if special?
|
286
|
+
|
287
|
+
case special
|
288
|
+
when :nan
|
289
|
+
n.ip = :indeterminate
|
290
|
+
when :inf
|
291
|
+
if sign=='-'
|
292
|
+
n.ip = :posinfinity
|
293
|
+
else
|
294
|
+
n.ip :neginfinity
|
295
|
+
end
|
296
|
+
else
|
297
|
+
n = nil
|
298
|
+
end
|
299
|
+
|
300
|
+
else
|
301
|
+
if dec_pos<=0
|
302
|
+
n.ip = 0
|
303
|
+
n.d = text_to_digits(dig_char(0)*(-dec_pos) + digits)
|
304
|
+
elsif dec_pos >= digits.length
|
305
|
+
n.ip = digits.to_i(@base)
|
306
|
+
if rep_pos<dec_pos
|
307
|
+
i=0
|
308
|
+
(dec_pos-digits.length).times do
|
309
|
+
n.ip *= @base
|
310
|
+
n.ip += @dgs.digit_value(digits[rep_pos+i]) if rep_pos+i<digits.length
|
311
|
+
i += 1
|
312
|
+
i=0 if i>=digits.length-rep_pos
|
313
|
+
end
|
314
|
+
n.d = []
|
315
|
+
while i<digits.length-rep_pos
|
316
|
+
n.d << @dgs.digit_value(digits[rep_pos+i])
|
317
|
+
i += 1
|
318
|
+
end
|
319
|
+
new_rep_pos = n.d.size + dec_pos
|
320
|
+
n.d += text_to_digits(digits[rep_pos..-1])
|
321
|
+
self.rep_pos = new_rep_pos
|
322
|
+
else
|
323
|
+
n.ip *= @base**(dec_pos-digits.length)
|
324
|
+
n.d = []
|
325
|
+
end
|
326
|
+
else
|
327
|
+
n.ip = digits[0...dec_pos].to_i(@base)
|
328
|
+
n.d = text_to_digits(digits[dec_pos..-1])
|
329
|
+
if rep_pos<dec_pos
|
330
|
+
new_rep_pos = n.d.size + dec_pos
|
331
|
+
n.d += text_to_digits(digits[rep_pos..-1])
|
332
|
+
self.rep_pos = new_rep_pos
|
333
|
+
puts "--rep_pos=#{rep_pos}"
|
334
|
+
end
|
335
|
+
end
|
336
|
+
n.sign = -1 if sign=='-'
|
337
|
+
n.rep_i = rep_pos - dec_pos
|
338
|
+
end
|
339
|
+
n.normalize!(!inexact) # keep trailing zeros for inexact numbers
|
340
|
+
return n
|
341
|
+
end
|
342
|
+
protected
|
343
|
+
def text_to_digits(txt)
|
344
|
+
#txt.split('').collect{|c| @dgs.digit_value(c)}
|
345
|
+
ds = []
|
346
|
+
txt.each_byte{|b| ds << @dgs.digit_value(b)}
|
347
|
+
ds
|
348
|
+
end
|
349
|
+
end
|
350
|
+
|
351
|
+
class RepDec
|
352
|
+
public
|
353
|
+
def to_NeutralNum(base_dgs=nil)
|
354
|
+
num = NeutralNum.new
|
355
|
+
if !ip.is_a?(Integer)
|
356
|
+
|
357
|
+
case ip
|
358
|
+
when :indeterminate
|
359
|
+
num.set_special :nan
|
360
|
+
when :posinfinity
|
361
|
+
num.set_special :inf,'+'
|
362
|
+
when :neginfinity
|
363
|
+
num.set_special :inf,'-'
|
364
|
+
else
|
365
|
+
num = nil
|
366
|
+
end
|
367
|
+
|
368
|
+
else
|
369
|
+
base_dgs ||= DigitsDef.base(@radix)
|
370
|
+
# assert base_dgs.radix == @radix
|
371
|
+
signch = sign<0 ? '-' : '+'
|
372
|
+
decimals = ip.to_s(@radix)
|
373
|
+
dec_pos = decimals.length
|
374
|
+
d.each {|dig| decimals << base_dgs.digit_char(dig) }
|
375
|
+
rep_pos = rep_i==nil ? decimals.length : dec_pos + rep_i
|
376
|
+
num.set signch, decimals, dec_pos, rep_pos, base_dgs
|
377
|
+
end
|
378
|
+
return num
|
379
|
+
end
|
380
|
+
end
|
381
|
+
|
382
|
+
# A Fmt object defines a numeric format.
|
383
|
+
#
|
384
|
+
# The formatting aspects managed by Fmt are:
|
385
|
+
# * mode and precision
|
386
|
+
# - #mode() and #orec() set the main paramters
|
387
|
+
# - see also #show_all_digits(), #approx_mode(), #insignificant_digits(),
|
388
|
+
# #sci_digits() and #show_plus()
|
389
|
+
# * separators
|
390
|
+
# - see #sep() and #grouping()
|
391
|
+
# * field justfification
|
392
|
+
# - #width() and the shortcut #pad0s()
|
393
|
+
# * numerical base
|
394
|
+
# - #base()
|
395
|
+
# * repeating numerals
|
396
|
+
# - #rep()
|
397
|
+
#
|
398
|
+
# Note that for every aspect there are also corresponding _mutator_
|
399
|
+
# methos (its name ending with a bang) that modify an object in place,
|
400
|
+
# instead of returning an altered copy.
|
401
|
+
#
|
402
|
+
# This class also contains class methods for numeric conversion:
|
403
|
+
# * Fmt.convert
|
404
|
+
# and for default and other predefined formats:
|
405
|
+
# * Fmt.default / Fmt.default=
|
406
|
+
# * Fmt.[] / Fmt.[]=
|
407
|
+
#
|
408
|
+
# The actual formatted reading and writting if performed by
|
409
|
+
# * #nio_write() (Nio::Formattable#nio_write)
|
410
|
+
# * #nio_read() (Nio::Formattable::ClassMethods#nio_read)
|
411
|
+
# Finally numerical objects can be rounded according to a format:
|
412
|
+
# * #nio_round() (Nio::Formattable#nio_round)
|
413
|
+
class Fmt
|
414
|
+
include StateEquivalent
|
415
|
+
|
416
|
+
class Error < StandardError # :nodoc:
|
417
|
+
end
|
418
|
+
class InvalidOption < Error # :nodoc:
|
419
|
+
end
|
420
|
+
class InvalidFormat < Error # :nodoc:
|
421
|
+
end
|
422
|
+
|
423
|
+
@@default_rounding_mode = :even
|
424
|
+
def initialize()
|
425
|
+
|
426
|
+
@dec_sep = '.'
|
427
|
+
@grp_sep = ','
|
428
|
+
@grp = []
|
429
|
+
|
430
|
+
@ndig = :exact
|
431
|
+
@mode=:gen
|
432
|
+
@round=Fmt.default_rounding_mode
|
433
|
+
@all_digits = false
|
434
|
+
@approx = :only_sig
|
435
|
+
@non_sig = '' # marker for insignificant digits of inexact values e.g. '#','0'
|
436
|
+
@sci_format = 1 # number of integral digits in the mantissa: -1 for all
|
437
|
+
|
438
|
+
@show_plus = false
|
439
|
+
|
440
|
+
@rep_begin = '<'
|
441
|
+
@rep_end = '>'
|
442
|
+
@rep_auto = '...'
|
443
|
+
@rep_n = 2
|
444
|
+
@rep_in = true
|
445
|
+
|
446
|
+
@width = 0
|
447
|
+
@fill_char = ' '
|
448
|
+
@adjust=:right
|
449
|
+
|
450
|
+
@base_radix = 10
|
451
|
+
@base_uppercase = true
|
452
|
+
@base_digits = DigitsDef.base(@base_radix, !@base_uppercase)
|
453
|
+
@show_base = false
|
454
|
+
@base_indicators = { 2=>'b', 8=>'o', 10=>'', 16=>'h', 0=>'r'} # 0: generic (used with radix)
|
455
|
+
@base_prefix = false
|
456
|
+
|
457
|
+
@nan_txt = 'NAN'
|
458
|
+
@inf_txt = 'Infinity'
|
459
|
+
|
460
|
+
yield self if block_given?
|
461
|
+
end
|
462
|
+
|
463
|
+
# Defines the separators used in numerals. This is relevant to
|
464
|
+
# both input and output.
|
465
|
+
#
|
466
|
+
# The first argument is the radix point separator (usually
|
467
|
+
# a point or a comma; by default it is a point.)
|
468
|
+
#
|
469
|
+
# The second argument is the group separator.
|
470
|
+
#
|
471
|
+
# Finally, the third argument is an array that defines the groups
|
472
|
+
# of digits to separate.
|
473
|
+
# By default it's [], which means that no grouping will be produced on output
|
474
|
+
# (but the group separator defined will be ignored in input.)
|
475
|
+
# To produce the common thousands separation a value of [3] must be passed,
|
476
|
+
# which means that groups of 3 digits are used.
|
477
|
+
def sep(dec_sep,grp_sep=nil,grp=nil)
|
478
|
+
dup.sep!(dec_sep,grp_sep,grp)
|
479
|
+
end
|
480
|
+
# This is the mutator version of #sep().
|
481
|
+
def sep!(dec_sep,grp_sep=nil,grp=nil)
|
482
|
+
set! :dec_sep=>dec_sep, :grp_sep=>grp_sep, :grp=>grp
|
483
|
+
end
|
484
|
+
|
485
|
+
# This defines the grouping of digits (which can also be defined in #sep()
|
486
|
+
def grouping(grp=[3],grp_sep=nil)
|
487
|
+
dup.grouping!(grp,grp_sep)
|
488
|
+
end
|
489
|
+
# This is the mutator version of #grouping().
|
490
|
+
def grouping!(grp=[3],grp_sep=nil)
|
491
|
+
set! :grp_sep=>grp_sep, :grp=>grp
|
492
|
+
end
|
493
|
+
|
494
|
+
# This is a shortcut to return a new default Fmt object
|
495
|
+
# and define the separators as with #sep().
|
496
|
+
def Fmt.sep(dec_sep,grp_sep=nil,grp=nil)
|
497
|
+
Fmt.default.sep(dec_sep,grp_sep,grp)
|
498
|
+
end
|
499
|
+
# This is a shortcut to return a new default Fmt object
|
500
|
+
# and define the grouping as with #grouping().
|
501
|
+
def Fmt.grouping(grp=[3],grp_sep=nil)
|
502
|
+
Fmt.default.grouping(grp,grp_sep)
|
503
|
+
end
|
504
|
+
|
505
|
+
# Define the formatting mode. There are two fixed parameters:
|
506
|
+
# - <tt>mode</tt> (only relevant for output)
|
507
|
+
# [<tt>:gen</tt>]
|
508
|
+
# (general) chooses automatically the shortes format
|
509
|
+
# [<tt>:fix</tt>]
|
510
|
+
# (fixed precision) is a simple format with a fixed number of digits
|
511
|
+
# after the point
|
512
|
+
# [<tt>:sig</tt>]
|
513
|
+
# (significance precision) is like :fix but using significant digits
|
514
|
+
# [<tt>:sci</tt>]
|
515
|
+
# (scientific) is the exponential form 1.234E2
|
516
|
+
# - <tt>precision</tt> (number of digits or :exact, only used for output)
|
517
|
+
# [<tt>exact</tt>]
|
518
|
+
# means that as many digits as necessary to unambiguosly define the
|
519
|
+
# value are used; this is the default.
|
520
|
+
#
|
521
|
+
# Other paramters can be passed in a hash after <tt>precision</tt>
|
522
|
+
# - <tt>:round</tt> rounding mode applied to conversions
|
523
|
+
# (this is relevant for both input and output). It must be one of:
|
524
|
+
# [<tt>:inf</tt>]
|
525
|
+
# rounds towards infinite; 1.5 is rounded to 2, -1.5 to -2
|
526
|
+
# [<tt>:zero</tt>]
|
527
|
+
# rounds towards zero; 1.5 is rounded to 1, -1.5 to 2
|
528
|
+
# [<tt>:even</tt>]
|
529
|
+
# rounds to the nearest even digit 1.5 rounds to 2, 2.5 to 2
|
530
|
+
# - <tt>:approx</tt> approximate mode
|
531
|
+
# [<tt>:only_sig</tt>]
|
532
|
+
# (the default) treats the value as an approximation and only
|
533
|
+
# significant digits (those that cannot take an arbitrary value without
|
534
|
+
# changing the specified value) are shown.
|
535
|
+
# [<tt>:exact</tt>]
|
536
|
+
# the value is interpreted as exact, there's no distinction between
|
537
|
+
# significant and insignificant digits.
|
538
|
+
# [<tt>:simplify</tt>]
|
539
|
+
# the value is simplified, if possible to a simpler (rational) value.
|
540
|
+
# - <tt>:show_all_digits</tt> if true, this forces to show digits that
|
541
|
+
# would otherwise not be shown in the <tt>:gen</tt> format: trailing
|
542
|
+
# zeros of exact types or non-signficative digits of inexact types.
|
543
|
+
# - <tt>:nonsignficative_digits</tt> assigns a character to display
|
544
|
+
# insignificant digits, # by default
|
545
|
+
def mode(mode,precision=nil,options={})
|
546
|
+
dup.mode!(mode,precision,options)
|
547
|
+
end
|
548
|
+
# This is the mutator version of #mode().
|
549
|
+
def mode!(mode,precision=nil,options={})
|
550
|
+
set! options.merge(:mode=>mode, :ndig=>precision)
|
551
|
+
end
|
552
|
+
|
553
|
+
# Defines the formatting mode like #mode() but using a different
|
554
|
+
# order of the first two parameters parameters, which is useful
|
555
|
+
# to change the precision only. Refer to #mode().
|
556
|
+
def prec(precision,mode=nil, options={})
|
557
|
+
dup.prec! precision, mode, options
|
558
|
+
end
|
559
|
+
# This is the mutator version of #prec().
|
560
|
+
def prec!(precision,mode=:gen, options={})
|
561
|
+
set! options.merge(:mode=>mode, :ndig=>precision)
|
562
|
+
end
|
563
|
+
|
564
|
+
# This is a shortcut to return a new default Fmt object
|
565
|
+
# and define the formatting mode as with #mode()
|
566
|
+
def Fmt.mode(mode,ndig=nil,options={})
|
567
|
+
Fmt.default.mode(mode,ndig,options)
|
568
|
+
end
|
569
|
+
# This is a shortcut to return a new default Fmt object
|
570
|
+
# and define the formatting mode as with #prec()
|
571
|
+
def Fmt.prec(ndig,mode=nil,options={})
|
572
|
+
Fmt.default.prec(ndig,mode,options)
|
573
|
+
end
|
574
|
+
|
575
|
+
# Rounding mode used when not specified otherwise
|
576
|
+
def Fmt.default_rounding_mode
|
577
|
+
@@default_rounding_mode
|
578
|
+
end
|
579
|
+
# The default rounding can be changed here; it starts with the value :even.
|
580
|
+
# See the rounding modes available in the description of method #mode().
|
581
|
+
def Fmt.default_rounding_mode=(m)
|
582
|
+
@@default_rounding_mode=m
|
583
|
+
Fmt.default = Fmt.default.round(m)
|
584
|
+
end
|
585
|
+
|
586
|
+
# This controls the display of the digits that are not necessary
|
587
|
+
# to specify the value unambiguosly (e.g. trailing zeros).
|
588
|
+
#
|
589
|
+
# The true (default) value forces the display of the requested number of digits
|
590
|
+
# and false will display only necessary digits.
|
591
|
+
def show_all_digits(ad=true)
|
592
|
+
dup.show_all_digits! ad
|
593
|
+
end
|
594
|
+
# This is the mutator version of #show_all_digits().
|
595
|
+
def show_all_digits!(ad=true)
|
596
|
+
set! :all_digits=>ad
|
597
|
+
end
|
598
|
+
# This defines the approximate mode (:only_sig, :exact, :simplify)
|
599
|
+
# just like the last parameter of #mode()
|
600
|
+
def approx_mode(mode)
|
601
|
+
dup.approx_mode! mode
|
602
|
+
end
|
603
|
+
# This is the mutator version of #approx_mode().
|
604
|
+
def approx_mode!(mode)
|
605
|
+
set! :approx=>mode
|
606
|
+
end
|
607
|
+
# Defines a character to stand for insignificant digits when
|
608
|
+
# a specific number of digits has been requested greater than then
|
609
|
+
# number of significant digits (for approximate types).
|
610
|
+
def insignificant_digits(ch='#')
|
611
|
+
dup.insignificant_digits! ch
|
612
|
+
end
|
613
|
+
# This is the mutator version of #insignificant_digits().
|
614
|
+
def insignificant_digits!(ch='#')
|
615
|
+
ch ||= ''
|
616
|
+
set! :non_sig=>ch
|
617
|
+
end
|
618
|
+
# Defines the number of significan digits before the radix separator
|
619
|
+
# in scientific notation. A negative value will set all significant digits
|
620
|
+
# before the radix separator. The special value <tt>:eng</tt> activates
|
621
|
+
# _engineering_ mode, in which the exponents are multiples of 3.
|
622
|
+
#
|
623
|
+
# For example:
|
624
|
+
# 0.1234.nio_write(Fmt.mode(:sci,4).sci_digits(0) -> 0.1234E0
|
625
|
+
# 0.1234.nio_write(Fmt.mode(:sci,4).sci_digits(3) -> 123.4E-3
|
626
|
+
# 0.1234.nio_write(Fmt.mode(:sci,4).sci_digits(-1) -> 1234.E-4
|
627
|
+
# 0.1234.nio_write(Fmt.mode(:sci,4).sci_digits(:eng) -> 123.4E-3
|
628
|
+
def sci_digits(n=-1)
|
629
|
+
dup.sci_digits! n
|
630
|
+
end
|
631
|
+
# This is the mutator version of #sci_digits().
|
632
|
+
def sci_digits!(n=-1)
|
633
|
+
set! :sci_format=>n
|
634
|
+
end
|
635
|
+
|
636
|
+
# This is a shortcut to return a new default Fmt object
|
637
|
+
# and define show_all_digits
|
638
|
+
def Fmt.show_all_digits(v=true)
|
639
|
+
Fmt.default.show_all_digits(v)
|
640
|
+
end
|
641
|
+
# This is a shortcut to return a new default Fmt object
|
642
|
+
# and define approx_mode
|
643
|
+
def Fmt.approx_mode(v)
|
644
|
+
Fmt.default.approx_mode(v)
|
645
|
+
end
|
646
|
+
# This is a shortcut to return a new default Fmt object
|
647
|
+
# and define insignificant digits
|
648
|
+
def Fmt.insignificant_digits(v='#')
|
649
|
+
Fmt.default.insignificant_digits(v)
|
650
|
+
end
|
651
|
+
# This is a shortcut to return a new default Fmt object
|
652
|
+
# and define sci_digits
|
653
|
+
def Fmt.sci_digits(v=-1)
|
654
|
+
Fmt.default.sci_digits(v)
|
655
|
+
end
|
656
|
+
|
657
|
+
# Controls the display of the sign for positive numbers
|
658
|
+
def show_plus(sp=true)
|
659
|
+
dup.show_plus! sp
|
660
|
+
end
|
661
|
+
# This is the mutator version of #show_plus().
|
662
|
+
def show_plus!(sp=true)
|
663
|
+
set! :show_plus=>sp
|
664
|
+
end
|
665
|
+
|
666
|
+
# This is a shortcut to return a new default Fmt object
|
667
|
+
# and define show_plus
|
668
|
+
def Fmt.show_plus(v=true)
|
669
|
+
Fmt.default.show_plus(v)
|
670
|
+
end
|
671
|
+
|
672
|
+
# Defines the handling and notation for repeating numerals. The parameters
|
673
|
+
# can be passed in order or in a hash:
|
674
|
+
# [<tt>:begin</tt>] is the beginning delimiter of repeating section (<)
|
675
|
+
# [<tt>:end</tt>] is the ending delimiter of repeating section (<)
|
676
|
+
# [<tt>:suffix</tt>] is the suffix used to indicate a implicit repeating decimal
|
677
|
+
# [<tt>:rep</tt>]
|
678
|
+
# if this parameter is greater than zero, on output the repeating section
|
679
|
+
# is repeated the indicated number of times followed by the suffix;
|
680
|
+
# otherwise the delimited notation is used.
|
681
|
+
# [<tt>:read</tt>]
|
682
|
+
# (true/false) determines if repeating decimals are
|
683
|
+
# recognized on input (true)
|
684
|
+
def rep(*params)
|
685
|
+
dup.rep!(*params)
|
686
|
+
end
|
687
|
+
# This is the mutator version of #rep().
|
688
|
+
def rep!(*params)
|
689
|
+
|
690
|
+
params << {} if params.size==0
|
691
|
+
if params[0].kind_of?(Hash)
|
692
|
+
params = params[0]
|
693
|
+
else
|
694
|
+
begch,endch,autoch,rep,read = *params
|
695
|
+
params = {:begin=>begch,:end=>endch,:suffix=>autoch,:nreps=>rep,:read=>read}
|
696
|
+
end
|
697
|
+
|
698
|
+
set! params
|
699
|
+
end
|
700
|
+
|
701
|
+
# This is a shortcut to return a new default Fmt object
|
702
|
+
# and define the repeating decimals mode as with #rep()
|
703
|
+
def Fmt.rep(*params)
|
704
|
+
Fmt.default.rep(*params)
|
705
|
+
end
|
706
|
+
|
707
|
+
# Sets the justificaton width, mode and fill character
|
708
|
+
#
|
709
|
+
# The mode accepts these values:
|
710
|
+
# [<tt>:right</tt>] (the default) justifies to the right (adds padding at the left)
|
711
|
+
# [<tt>:left</tt>] justifies to the left (adds padding to the right)
|
712
|
+
# [<tt>:internal</tt>] like :right, but the sign is kept to the left, outside the padding.
|
713
|
+
# [<tt>:center</tt>] centers the number in the field
|
714
|
+
def width(w,adj=nil,ch=nil)
|
715
|
+
dup.width! w,adj,ch
|
716
|
+
end
|
717
|
+
# This is the mutator version of #width().
|
718
|
+
def width!(w,adj=nil,ch=nil)
|
719
|
+
set! :width=>w, :adjust=>adj, :fill_char=>ch
|
720
|
+
end
|
721
|
+
# Defines the justification (as #width()) with the given
|
722
|
+
# width, internal mode and filling with zeros.
|
723
|
+
#
|
724
|
+
# Note that if you also use grouping separators, the filling 0s
|
725
|
+
# will not be separated.
|
726
|
+
def pad0s(w)
|
727
|
+
dup.pad0s! w
|
728
|
+
end
|
729
|
+
# This is the mutator version of #pad0s().
|
730
|
+
def pad0s!(w)
|
731
|
+
width! w, :internal, '0'
|
732
|
+
end
|
733
|
+
# This is a shortcut to create a new Fmt object and define the width
|
734
|
+
# parameters as with #widht()
|
735
|
+
def Fmt.width(w,adj=nil,ch=nil)
|
736
|
+
Fmt.default.width(w,adj,ch)
|
737
|
+
end
|
738
|
+
# This is a shortcut to create a new Fmt object and define numeric
|
739
|
+
# padding as with #pad0s()
|
740
|
+
def Fmt.pad0s(w)
|
741
|
+
Fmt.default.pad0s(w)
|
742
|
+
end
|
743
|
+
|
744
|
+
# defines the numerical base; the second parameters forces the use
|
745
|
+
# of uppercase letters for bases greater than 10.
|
746
|
+
def base(b, uppercase=nil)
|
747
|
+
dup.base! b, uppercase
|
748
|
+
end
|
749
|
+
# This is the mutator version of #base().
|
750
|
+
def base!(b, uppercase=nil)
|
751
|
+
set! :base_radix=>b, :base_uppercase=>uppercase
|
752
|
+
end
|
753
|
+
# This is a shortcut to create a new Fmt object and define the base
|
754
|
+
def Fmt.base(b, uppercase=nil)
|
755
|
+
Fmt.default.base(b, uppercase)
|
756
|
+
end
|
757
|
+
# returns the exponent char used with the specified base
|
758
|
+
def get_exp_char(base) # :nodoc:
|
759
|
+
base ||= @base_radix
|
760
|
+
base<=10 ? 'E' : '^'
|
761
|
+
end
|
762
|
+
|
763
|
+
# returns the base
|
764
|
+
def get_base # :nodoc:
|
765
|
+
@base_radix
|
766
|
+
end
|
767
|
+
# returns the digit characters used for a base
|
768
|
+
def get_base_digits(b=nil) # :nodoc:
|
769
|
+
(b.nil? || b==@base_radix) ? @base_digits : DigitsDef.base(b,!@base_uppercase)
|
770
|
+
end
|
771
|
+
# returns true if uppercase digits are used
|
772
|
+
def get_base_uppercase? # :nodoc:
|
773
|
+
@base_uppercase
|
774
|
+
end
|
775
|
+
|
776
|
+
# returns the formatting mode
|
777
|
+
def get_mode # :nodoc:
|
778
|
+
@mode
|
779
|
+
end
|
780
|
+
# returns the precision (number of digits)
|
781
|
+
def get_ndig # :nodoc:
|
782
|
+
@ndig
|
783
|
+
end
|
784
|
+
# return the show_all_digits state
|
785
|
+
def get_all_digits? # :nodoc:
|
786
|
+
@all_digits
|
787
|
+
end
|
788
|
+
# returns the approximate mode
|
789
|
+
def get_approx # :nodoc:
|
790
|
+
@approx
|
791
|
+
end
|
792
|
+
|
793
|
+
# returns the rounding mode
|
794
|
+
def get_round # :nodoc:
|
795
|
+
@round
|
796
|
+
end
|
797
|
+
|
798
|
+
# Method used internally to format a neutral numeral
|
799
|
+
def nio_write_formatted(neutral) # :nodoc:
|
800
|
+
str = ''
|
801
|
+
if neutral.special?
|
802
|
+
str << neutral.sign
|
803
|
+
case neutral.special
|
804
|
+
when :inf
|
805
|
+
str << @inf_txt
|
806
|
+
when :nan
|
807
|
+
str << @nan_txt
|
808
|
+
end
|
809
|
+
else
|
810
|
+
zero = get_base_digits(neutral.base).digit_char(0).chr
|
811
|
+
neutral = neutral.dup
|
812
|
+
round! neutral
|
813
|
+
if neutral.zero?
|
814
|
+
str << neutral.sign if neutral.sign=='-' # show - if number was <0 before rounding
|
815
|
+
str << zero
|
816
|
+
if @ndig.kind_of?(Numeric) && @ndig>0 && @mode==:fix
|
817
|
+
str << @dec_sep << zero*@ndig
|
818
|
+
end
|
819
|
+
else
|
820
|
+
|
821
|
+
neutral.trimLeadZeros
|
822
|
+
actual_mode = @mode
|
823
|
+
trim_trail_zeros = !@all_digits # false
|
824
|
+
|
825
|
+
integral_digits = @sci_format
|
826
|
+
if integral_digits == :eng
|
827
|
+
integral_digits = 1
|
828
|
+
while (neutral.dec_pos - integral_digits).modulo(3) != 0
|
829
|
+
integral_digits += 1
|
830
|
+
end
|
831
|
+
elsif integral_digits==:all || integral_digits < 0
|
832
|
+
if neutral.inexact? && @non_sig!='' && @ndig.kind_of?(Numeric)
|
833
|
+
integral_digits = @ndig
|
834
|
+
else
|
835
|
+
integral_digits = neutral.digits.length
|
836
|
+
end
|
837
|
+
end
|
838
|
+
exp = neutral.dec_pos - integral_digits
|
839
|
+
|
840
|
+
case actual_mode
|
841
|
+
when :gen # general (automatic)
|
842
|
+
# @ndig means significant digits
|
843
|
+
actual_mode = :sig
|
844
|
+
actual_mode = :sci if use_scientific?(neutral, exp)
|
845
|
+
trim_trail_zeros = !@all_digits # true
|
846
|
+
end
|
847
|
+
|
848
|
+
case actual_mode
|
849
|
+
when :fix, :sig #, :gen
|
850
|
+
|
851
|
+
str << neutral.sign if @show_plus || neutral.sign!='+'
|
852
|
+
|
853
|
+
if @show_base && @base_prefix
|
854
|
+
b_prefix = @base_indicators[neutral.base]
|
855
|
+
str << b_prefix if b_prefix
|
856
|
+
end
|
857
|
+
|
858
|
+
if @ndig==:exact
|
859
|
+
neutral.sign = '+'
|
860
|
+
str << neutral.to_RepDec.getS(@rep_n, getRepDecOpt(neutral.base))
|
861
|
+
else
|
862
|
+
#zero = get_base_digits.digit_char(0).chr
|
863
|
+
ns_digits = ''
|
864
|
+
|
865
|
+
nd = neutral.digits.length
|
866
|
+
if actual_mode==:fix
|
867
|
+
nd -= neutral.dec_pos
|
868
|
+
end
|
869
|
+
if neutral.inexact? && @ndig>nd # assert no rep-dec.
|
870
|
+
ns_digits = @non_sig*(@ndig-nd)
|
871
|
+
end
|
872
|
+
|
873
|
+
digits = neutral.digits + ns_digits
|
874
|
+
if neutral.dec_pos<=0
|
875
|
+
str << zero+@dec_sep+zero*(-neutral.dec_pos) + digits
|
876
|
+
elsif neutral.dec_pos >= digits.length
|
877
|
+
str << group(digits + zero*(neutral.dec_pos-digits.length))
|
878
|
+
else
|
879
|
+
str << group(digits[0...neutral.dec_pos]) + @dec_sep + digits[neutral.dec_pos..-1]
|
880
|
+
end
|
881
|
+
end
|
882
|
+
|
883
|
+
#str = str.chomp(zero).chomp(@dec_sep) if trim_trail_zeros && str.include?(@dec_sep)
|
884
|
+
if trim_trail_zeros && str.include?(@dec_sep) && str[-@rep_auto.size..-1]!=@rep_auto
|
885
|
+
str.chop! while str[-1]==zero[0]
|
886
|
+
str.chomp!(@dec_sep)
|
887
|
+
#puts str
|
888
|
+
end
|
889
|
+
|
890
|
+
|
891
|
+
when :sci
|
892
|
+
|
893
|
+
str << neutral.sign if @show_plus || neutral.sign!='+'
|
894
|
+
|
895
|
+
if @show_base && @base_prefix
|
896
|
+
b_prefix = @base_indicators[neutral.base]
|
897
|
+
str << b_prefix if b_prefix
|
898
|
+
end
|
899
|
+
|
900
|
+
#zero = get_base_digits.digit_char(0).chr
|
901
|
+
if @ndig==:exact
|
902
|
+
neutral.sign = '+'
|
903
|
+
neutral.dec_pos-=exp
|
904
|
+
str << neutral.to_RepDec.getS(@rep_n, getRepDecOpt(neutral.base))
|
905
|
+
else
|
906
|
+
ns_digits = ''
|
907
|
+
|
908
|
+
nd = neutral.digits.length
|
909
|
+
if actual_mode==:fix
|
910
|
+
nd -= neutral.dec_pos
|
911
|
+
end
|
912
|
+
if neutral.inexact? && @ndig>nd # assert no rep-dec.
|
913
|
+
ns_digits = @non_sig*(@ndig-nd)
|
914
|
+
end
|
915
|
+
|
916
|
+
digits = neutral.digits + ns_digits
|
917
|
+
str << ((integral_digits<1) ? zero : digits[0...integral_digits])
|
918
|
+
str << @dec_sep
|
919
|
+
str << digits[integral_digits...@ndig]
|
920
|
+
pad_right =(@ndig+1-str.length)
|
921
|
+
str << zero*pad_right if pad_right>0 && !neutral.inexact? # maybe we didn't have enought digits
|
922
|
+
end
|
923
|
+
|
924
|
+
#str = str.chomp(zero).chomp(@dec_sep) if trim_trail_zeros && str.include?(@dec_sep)
|
925
|
+
if trim_trail_zeros && str.include?(@dec_sep) && str[-@rep_auto.size..-1]!=@rep_auto
|
926
|
+
str.chop! while str[-1]==zero[0]
|
927
|
+
str.chomp!(@dec_sep)
|
928
|
+
#puts str
|
929
|
+
end
|
930
|
+
|
931
|
+
str << get_exp_char(neutral.base)
|
932
|
+
str << exp.to_s
|
933
|
+
|
934
|
+
end
|
935
|
+
|
936
|
+
end
|
937
|
+
end
|
938
|
+
|
939
|
+
if @show_base && !@base_prefix
|
940
|
+
b_prefix = @base_indicators[neutral.base]
|
941
|
+
str << b_prefix if b_prefix
|
942
|
+
end
|
943
|
+
|
944
|
+
|
945
|
+
if @width>0 && @fill_char!=''
|
946
|
+
l = @width - str.length
|
947
|
+
if l>0
|
948
|
+
case @adjust
|
949
|
+
when :internal
|
950
|
+
sign = ''
|
951
|
+
if str[0,1]=='+' || str[0,1]=='-'
|
952
|
+
sign = str[0,1]
|
953
|
+
str = str[1...str.length]
|
954
|
+
end
|
955
|
+
str = sign + @fill_char*l + str
|
956
|
+
when :center
|
957
|
+
str = @fill_char*(l/2) + str + @fill_char*(l-l/2)
|
958
|
+
when :right
|
959
|
+
str = @fill_char*l + str
|
960
|
+
when :left
|
961
|
+
str = str + @fill_char*l
|
962
|
+
end
|
963
|
+
end
|
964
|
+
end
|
965
|
+
|
966
|
+
return str
|
967
|
+
end
|
968
|
+
|
969
|
+
# round a neutral numeral according to the format options
|
970
|
+
def round!(neutral) # :nodoc:
|
971
|
+
neutral.round! @ndig, @mode, @round
|
972
|
+
end
|
973
|
+
|
974
|
+
@@sci_fmt = nil
|
975
|
+
|
976
|
+
def nio_read_formatted(txt) # :nodoc:
|
977
|
+
txt = txt.dup
|
978
|
+
num = nil
|
979
|
+
|
980
|
+
base = nil
|
981
|
+
|
982
|
+
base ||= get_base
|
983
|
+
|
984
|
+
zero = get_base_digits(base).digit_char(0).chr
|
985
|
+
txt.tr!(@non_sig,zero) # we don't simply remove it because it may be before the radix point
|
986
|
+
|
987
|
+
exp = 0
|
988
|
+
x_char = get_exp_char(base)
|
989
|
+
|
990
|
+
exp_i = txt.index(x_char)
|
991
|
+
exp_i = txt.index(x_char.downcase) if exp_i===nil
|
992
|
+
if exp_i!=nil
|
993
|
+
exp = txt[exp_i+1...txt.length].to_i
|
994
|
+
txt = txt[0...exp_i]
|
995
|
+
end
|
996
|
+
|
997
|
+
|
998
|
+
opt = getRepDecOpt(base)
|
999
|
+
if @rep_in
|
1000
|
+
#raise InvalidFormat,"Invalid numerical base" if base!=10
|
1001
|
+
rd = RepDec.new # get_base not necessary: setS sets it from options
|
1002
|
+
rd.setS txt, opt
|
1003
|
+
num = rd.to_NeutralNum(opt.digits)
|
1004
|
+
else
|
1005
|
+
# to do: use RepDec.parse; then build NeutralNum directly
|
1006
|
+
opt.set_delim '',''
|
1007
|
+
opt.set_suffix ''
|
1008
|
+
rd = RepDec.new # get_base not necessary: setS sets it from options
|
1009
|
+
rd.setS txt, opt
|
1010
|
+
num = rd.to_NeutralNum(opt.digits)
|
1011
|
+
end
|
1012
|
+
num.rounding = get_round
|
1013
|
+
num.dec_pos += exp
|
1014
|
+
return num
|
1015
|
+
end
|
1016
|
+
|
1017
|
+
|
1018
|
+
@@fmts = {
|
1019
|
+
:def=>Fmt.new.freeze
|
1020
|
+
}
|
1021
|
+
# Returns the current default format.
|
1022
|
+
def self.default
|
1023
|
+
d = self[:def]
|
1024
|
+
if block_given?
|
1025
|
+
d = d.dup
|
1026
|
+
yield d
|
1027
|
+
end
|
1028
|
+
d
|
1029
|
+
end
|
1030
|
+
# Defines the current default format.
|
1031
|
+
def self.default=(fmt)
|
1032
|
+
self[:def] = fmt
|
1033
|
+
end
|
1034
|
+
# Assigns a format to a name in the formats repository.
|
1035
|
+
def self.[]=(tag,fmt_def)
|
1036
|
+
@@fmts[tag.to_sym]=fmt_def.freeze
|
1037
|
+
end
|
1038
|
+
# Retrieves a named format from the repository.
|
1039
|
+
def self.[](tag)
|
1040
|
+
@@fmts[tag.to_sym]
|
1041
|
+
end
|
1042
|
+
|
1043
|
+
protected
|
1044
|
+
|
1045
|
+
@@valid_properties = nil
|
1046
|
+
ALIAS_PROPERTIES = {
|
1047
|
+
:show_all_digits=>:all_digits,
|
1048
|
+
:rounding_mode=>:round,
|
1049
|
+
:approx_mode=>:approx,
|
1050
|
+
:sci_digits=>:sci_format,
|
1051
|
+
:non_signitificative_digits=>:non_sig,
|
1052
|
+
:begin=>:rep_begin,
|
1053
|
+
:end=>:rep_end,
|
1054
|
+
:suffix=>:rep_auto,
|
1055
|
+
:nreps=>:rep_n,
|
1056
|
+
:read=>:rep_in
|
1057
|
+
}
|
1058
|
+
def set!(properties={}) # :nodoc:
|
1059
|
+
|
1060
|
+
|
1061
|
+
@@valid_properties ||= instance_variables.collect{|v| v[1..-1].to_sym}
|
1062
|
+
|
1063
|
+
|
1064
|
+
properties.each do |k,v|
|
1065
|
+
al = ALIAS_PROPERTIES[k]
|
1066
|
+
if al
|
1067
|
+
properties[al] = v
|
1068
|
+
properties.delete k
|
1069
|
+
elsif !@@valid_properties.include?(k)
|
1070
|
+
raise InvalidOption, "Invalid option: #{k}"
|
1071
|
+
end
|
1072
|
+
end
|
1073
|
+
|
1074
|
+
|
1075
|
+
if properties[:grp_sep].nil? && !properties[:dec_sep].nil? && properties[:dec_sep]!=@dec_sep && properties[:dec_sep]==@grp_sep
|
1076
|
+
properties[:grp_sep] = properties[:dec_sep]=='.' ? ',' : '.'
|
1077
|
+
end
|
1078
|
+
|
1079
|
+
if properties[:all_digits].nil? && (properties[:ndig] || properties[:mode])
|
1080
|
+
ndig = properties[:ndig] || @ndig
|
1081
|
+
mode = properties[:mode] || @mode
|
1082
|
+
properties[:all_digits] = ndig!=:exact && mode!=:gen
|
1083
|
+
end
|
1084
|
+
|
1085
|
+
if !properties[:all_digits].nil? && properties[:non_sig].nil?
|
1086
|
+
properties[:non_sig] = '' unless properties[:all_digits]
|
1087
|
+
elsif !properties[:non_sig].nil? && properties[:all_digits].nil?
|
1088
|
+
properties[:all_digits] = true if properties[:non_sig]!=''
|
1089
|
+
end
|
1090
|
+
|
1091
|
+
if !properties[:base_radix].nil? || !properties[:base_uppercase].nil?
|
1092
|
+
base = properties[:base_radix] || @base_radix
|
1093
|
+
uppercase = properties[:base_uppercase] || @base_uppercase
|
1094
|
+
properties[:base_digits] = DigitsDef.base(base, !uppercase)
|
1095
|
+
end
|
1096
|
+
|
1097
|
+
|
1098
|
+
properties.each do |k,v|
|
1099
|
+
instance_variable_set "@#{k}", v unless v.nil?
|
1100
|
+
end
|
1101
|
+
|
1102
|
+
self
|
1103
|
+
end
|
1104
|
+
|
1105
|
+
def set(properties={}) # :nodoc:
|
1106
|
+
self.dup.set!(properties)
|
1107
|
+
end
|
1108
|
+
|
1109
|
+
def use_scientific?(neutral,exp) # :nodoc:
|
1110
|
+
nd = @ndig.kind_of?(Numeric) ? @ndig : [neutral.digits.length,10].max
|
1111
|
+
if @@sci_fmt==:hp
|
1112
|
+
puts " #{nd} ndpos=#{neutral.dec_pos} ndlen=#{neutral.digits.length}"
|
1113
|
+
neutral.dec_pos>nd || ([neutral.digits.length,nd].min-neutral.dec_pos)>nd
|
1114
|
+
else
|
1115
|
+
exp<-4 || exp>=nd
|
1116
|
+
end
|
1117
|
+
end
|
1118
|
+
|
1119
|
+
def getRepDecOpt(base=nil) # :nodoc:
|
1120
|
+
rd_opt = RepDec::Opt.new
|
1121
|
+
rd_opt.begin_rep = @rep_begin
|
1122
|
+
rd_opt.end_rep = @rep_end
|
1123
|
+
rd_opt.auto_rep = @rep_auto
|
1124
|
+
rd_opt.dec_sep = @dec_sep
|
1125
|
+
rd_opt.grp_sep = @grp_sep
|
1126
|
+
rd_opt.grp = @grp
|
1127
|
+
rd_opt.inf_txt = @inf_txt
|
1128
|
+
rd_opt.nan_txt = @nan_txt
|
1129
|
+
rd_opt.set_digits(get_base_digits(base))
|
1130
|
+
# if base && (base != get_base_digits.radix)
|
1131
|
+
# rd_opt.set_digits(get_base_digits(base))
|
1132
|
+
# else
|
1133
|
+
# rd_opt.set_digits get_base_digits
|
1134
|
+
# end
|
1135
|
+
return rd_opt
|
1136
|
+
end
|
1137
|
+
|
1138
|
+
def group(digits) # :nodoc:
|
1139
|
+
RepDec.group_digits(digits, getRepDecOpt)
|
1140
|
+
end
|
1141
|
+
|
1142
|
+
end
|
1143
|
+
|
1144
|
+
# This is a mix-in module to add formatting capabilities no numerical classes.
|
1145
|
+
# A class that includes this module should provide the methods
|
1146
|
+
# nio_write_neutral(fmt):: an instance method to write the value to
|
1147
|
+
# a neutral numeral. The format is passed so that
|
1148
|
+
# the base, for example, is available.
|
1149
|
+
# nio_read_neutral(neutral):: a class method to create a value from a neutral
|
1150
|
+
# numeral.
|
1151
|
+
module Formattable
|
1152
|
+
|
1153
|
+
# This is the method available in all formattable objects
|
1154
|
+
# to format the value into a text string according
|
1155
|
+
# to the optional format passed.
|
1156
|
+
def nio_write(fmt=Fmt.default)
|
1157
|
+
neutral = nio_write_neutral(fmt)
|
1158
|
+
fmt.nio_write_formatted(neutral)
|
1159
|
+
end
|
1160
|
+
|
1161
|
+
module ClassMethods
|
1162
|
+
# This is the method available in all formattable clases
|
1163
|
+
# to read a formatted value from a text string into
|
1164
|
+
# a value the class, according to the optional format passed.
|
1165
|
+
def nio_read(txt,fmt=Fmt.default)
|
1166
|
+
neutral = fmt.nio_read_formatted(txt)
|
1167
|
+
nio_read_neutral neutral
|
1168
|
+
end
|
1169
|
+
end
|
1170
|
+
|
1171
|
+
# Round a formattable object according to the rounding mode and
|
1172
|
+
# precision of a format.
|
1173
|
+
def nio_round(fmt=Fmt.default)
|
1174
|
+
neutral = nio_write_neutral(fmt)
|
1175
|
+
fmt.round! neutral
|
1176
|
+
self.class.nio_read_neutral neutral
|
1177
|
+
end
|
1178
|
+
|
1179
|
+
def self.append_features(mod) # :nodoc:
|
1180
|
+
super
|
1181
|
+
mod.extend ClassMethods
|
1182
|
+
end
|
1183
|
+
|
1184
|
+
end
|
1185
|
+
|
1186
|
+
Fmt[:comma] = Fmt.sep(',','.')
|
1187
|
+
Fmt[:comma_th] = Fmt.sep(',','.',[3])
|
1188
|
+
Fmt[:dot] = Fmt.sep('.',',')
|
1189
|
+
Fmt[:dot_th] = Fmt.sep('.',',',[3])
|
1190
|
+
Fmt[:code] = Fmt.new.prec(20) # don't use :exact to avoid repeating numerals
|
1191
|
+
|
1192
|
+
class Fmt
|
1193
|
+
# Intermediate conversion format for simplified conversion
|
1194
|
+
CONV_FMT = Fmt.prec(:exact).rep('<','>','...',0).approx_mode(:simplify)
|
1195
|
+
# Intermediate conversion format for exact conversion
|
1196
|
+
CONV_FMT_STRICT = Fmt.prec(:exact).rep('<','>','...',0).approx_mode(:exact)
|
1197
|
+
# Numerical conversion: converts the quantity +x+ to an object
|
1198
|
+
# of class +type+.
|
1199
|
+
#
|
1200
|
+
# The third parameter is the kind of conversion:
|
1201
|
+
# [<tt>:approx</tt>]
|
1202
|
+
# Tries to find an approximate simpler value if possible for inexact
|
1203
|
+
# numeric types. This is the default. This is slower in general and
|
1204
|
+
# may take some seconds in some cases.
|
1205
|
+
# [<tt>:exact</tt>]
|
1206
|
+
# Performs a conversion as exact as possible.
|
1207
|
+
# The third parameter is true for approximate
|
1208
|
+
# conversion (inexact values are simplified if possible) and false
|
1209
|
+
# for conversions as exact as possible.
|
1210
|
+
def Fmt.convert(x, type, mode=:approx)
|
1211
|
+
fmt = mode==:approx ? CONV_FMT : CONV_FMT_STRICT
|
1212
|
+
# return x.prec(type)
|
1213
|
+
if !(x.is_a?(type))
|
1214
|
+
# return type.nio_read(x.nio_write(fmt),fmt)
|
1215
|
+
|
1216
|
+
x = x.nio_write_neutral(fmt)
|
1217
|
+
x = type.nio_read_neutral(x)
|
1218
|
+
|
1219
|
+
end
|
1220
|
+
x
|
1221
|
+
end
|
1222
|
+
end
|
1223
|
+
|
1224
|
+
module_function
|
1225
|
+
|
1226
|
+
def nio_float_to_bigdecimal(x,prec) # :nodoc:
|
1227
|
+
if prec.nil?
|
1228
|
+
x = Nio.convert(x,BigDecimal,:approx)
|
1229
|
+
elsif prec==:exact
|
1230
|
+
x = Nio.convert(x,BigDecimal,:exact)
|
1231
|
+
else
|
1232
|
+
x = BigDecimal(x.nio_write(Nio::Fmt.new.prec(prec,:sig)))
|
1233
|
+
end
|
1234
|
+
x
|
1235
|
+
end
|
1236
|
+
|
1237
|
+
|
1238
|
+
module Clinger # :nodoc: all
|
1239
|
+
module_function
|
1240
|
+
|
1241
|
+
def algM(f,e,round_mode,eb=10,beta=Float::RADIX,n=Float::MANT_DIG,min_e=Float::MIN_EXP-Float::MANT_DIG,max_e=Float::MAX_EXP-Float::MANT_DIG)
|
1242
|
+
|
1243
|
+
if e<0
|
1244
|
+
u,v,k = f,eb**(-e),0
|
1245
|
+
else
|
1246
|
+
u,v,k = f*(eb**e),1,0
|
1247
|
+
end
|
1248
|
+
|
1249
|
+
loop do
|
1250
|
+
x = u.div(v)
|
1251
|
+
# overflow if k>=max_e
|
1252
|
+
if (x>=beta**(n-1) && x<beta**n) || k==min_e || k==max_e
|
1253
|
+
return ratio_float(u,v,k,round_mode,beta,n)
|
1254
|
+
elsif x<beta**(n-1)
|
1255
|
+
u *= beta
|
1256
|
+
k -= 1
|
1257
|
+
elsif x>=beta**n
|
1258
|
+
v *= beta
|
1259
|
+
k += 1
|
1260
|
+
end
|
1261
|
+
end
|
1262
|
+
|
1263
|
+
end
|
1264
|
+
|
1265
|
+
def ratio_float(u,v,k,round_mode,beta=Float::RADIX,n=Float::MANT_DIG)
|
1266
|
+
q = u.div v
|
1267
|
+
r = u-q*v
|
1268
|
+
v_r = v-r
|
1269
|
+
z = Math.ldexp(q,k)
|
1270
|
+
if r<v_r
|
1271
|
+
z
|
1272
|
+
elsif r>v_r
|
1273
|
+
nextfloat z
|
1274
|
+
elsif (round_mode==:even && q.even?) || (round_mode==:zero)
|
1275
|
+
z
|
1276
|
+
else
|
1277
|
+
nextfloat z
|
1278
|
+
end
|
1279
|
+
end
|
1280
|
+
|
1281
|
+
def nextfloat(x)
|
1282
|
+
f,e = Math.frexp(x)
|
1283
|
+
e = Float::MIN_EXP if f==0
|
1284
|
+
e = [Float::MIN_EXP,e].max
|
1285
|
+
dx = Math.ldexp(1,e-Float::MANT_DIG) #Math.ldexp(Math.ldexp(1.0,-Float::MANT_DIG),e)
|
1286
|
+
x + dx
|
1287
|
+
end
|
1288
|
+
|
1289
|
+
def prevfloat(x)
|
1290
|
+
f,e = Math.frexp(x)
|
1291
|
+
e = Float::MIN_EXP if f==0
|
1292
|
+
e = [Float::MIN_EXP,e].max
|
1293
|
+
dx = Math.ldexp(1,e-Float::MANT_DIG) #Math.ldexp(Math.ldexp(1.0,-Float::MANT_DIG),e)
|
1294
|
+
if e==Float::MIN_EXP || f!=0.5 #0.5==Math.ldexp(2**(bits-1),-Float::MANT_DIG)
|
1295
|
+
x - dx
|
1296
|
+
else
|
1297
|
+
x - dx/2 # x - Math.ldexp(Math.ldexp(1.0,-Float::MANT_DIG),e-1)
|
1298
|
+
end
|
1299
|
+
end
|
1300
|
+
|
1301
|
+
end
|
1302
|
+
|
1303
|
+
module BurgerDybvig # :nodoc: all
|
1304
|
+
module_function
|
1305
|
+
|
1306
|
+
def float_to_digits(v,f,e,round_mode,min_e,p,b,_B)
|
1307
|
+
|
1308
|
+
case round_mode
|
1309
|
+
when :even
|
1310
|
+
roundl = roundh = f.even?
|
1311
|
+
when :inf
|
1312
|
+
roundl = true
|
1313
|
+
roundh = false
|
1314
|
+
when :zero
|
1315
|
+
roundl = false
|
1316
|
+
roundh = true
|
1317
|
+
else
|
1318
|
+
# here we don't assume any rounding in the floating point numbers
|
1319
|
+
# the result is valid for any rounding
|
1320
|
+
roundl = false
|
1321
|
+
roundh = false
|
1322
|
+
end
|
1323
|
+
|
1324
|
+
if e >= 0
|
1325
|
+
if f != exptt(b,p-1)
|
1326
|
+
be = exptt(b,e)
|
1327
|
+
r,s,m_p,m_m,k = scale(f*be*2,2,be,be,0,_B,roundl ,roundh,v)
|
1328
|
+
else
|
1329
|
+
be = exptt(b,e)
|
1330
|
+
be1 = be*b
|
1331
|
+
r,s,m_p,m_m,k = scale(f*be1*2,b*2,be1,be,0,_B,roundl ,roundh,v)
|
1332
|
+
end
|
1333
|
+
else
|
1334
|
+
if e==min_e or f != exptt(b,p-1)
|
1335
|
+
r,s,m_p,m_m,k = scale(f*2,exptt(b,-e)*2,1,1,0,_B,roundl ,roundh,v)
|
1336
|
+
else
|
1337
|
+
r,s,m_p,m_m,k = scale(f*b*2,exptt(b,1-e)*2,b,1,0,_B,roundl ,roundh,v)
|
1338
|
+
end
|
1339
|
+
end
|
1340
|
+
[k]+generate(r,s,m_p,m_m,_B,roundl ,roundh)
|
1341
|
+
end
|
1342
|
+
|
1343
|
+
def scale(r,s,m_p,m_m,k,_B,low_ok ,high_ok,v)
|
1344
|
+
return scale2(r,s,m_p,m_m,k,_B,low_ok ,high_ok) if v==0
|
1345
|
+
est = (logB(_B,v)-1E-10).ceil.to_i
|
1346
|
+
if est>=0
|
1347
|
+
fixup(r,s*exptt(_B,est),m_p,m_m,est,_B,low_ok,high_ok)
|
1348
|
+
else
|
1349
|
+
sc = exptt(_B,-est)
|
1350
|
+
fixup(r*sc,s,m_p*sc,m_m*sc,est,_B,low_ok,high_ok)
|
1351
|
+
end
|
1352
|
+
end
|
1353
|
+
|
1354
|
+
def fixup(r,s,m_p,m_m,k,_B,low_ok,high_ok)
|
1355
|
+
if (high_ok ? (r+m_p >= s) : (r+m_p > s)) # too low?
|
1356
|
+
[r,s*_B,m_p,m_m,k+1]
|
1357
|
+
else
|
1358
|
+
[r,s,m_p,m_m,k]
|
1359
|
+
end
|
1360
|
+
end
|
1361
|
+
|
1362
|
+
def scale2(r,s,m_p,m_m,k,_B,low_ok ,high_ok)
|
1363
|
+
loop do
|
1364
|
+
if (high_ok ? (r+m_p >= s) : (r+m_p > s)) # k is too low
|
1365
|
+
s *= _B
|
1366
|
+
k += 1
|
1367
|
+
elsif (high_ok ? ((r+m_p)*_B<s) : ((r+m_p)*_B<=s)) # k is too high
|
1368
|
+
r *= _B
|
1369
|
+
m_p *= _B
|
1370
|
+
m_m *= _B
|
1371
|
+
k -= 1
|
1372
|
+
else
|
1373
|
+
break
|
1374
|
+
end
|
1375
|
+
end
|
1376
|
+
[r,s,m_p,m_m,k]
|
1377
|
+
end
|
1378
|
+
|
1379
|
+
def generate(r,s,m_p,m_m,_B,low_ok ,high_ok)
|
1380
|
+
list = []
|
1381
|
+
loop do
|
1382
|
+
d,r = (r*_B).divmod(s)
|
1383
|
+
m_p *= _B
|
1384
|
+
m_m *= _B
|
1385
|
+
tc1 = low_ok ? (r<=m_m) : (r<m_m)
|
1386
|
+
tc2 = high_ok ? (r+m_p >= s) : (r+m_p > s)
|
1387
|
+
|
1388
|
+
if not tc1
|
1389
|
+
if not tc2
|
1390
|
+
list << d
|
1391
|
+
else
|
1392
|
+
list << d+1
|
1393
|
+
break
|
1394
|
+
end
|
1395
|
+
else
|
1396
|
+
if not tc2
|
1397
|
+
list << d
|
1398
|
+
break
|
1399
|
+
else
|
1400
|
+
if r*2 < s
|
1401
|
+
list << d
|
1402
|
+
break
|
1403
|
+
else
|
1404
|
+
list << d+1
|
1405
|
+
break
|
1406
|
+
end
|
1407
|
+
end
|
1408
|
+
end
|
1409
|
+
|
1410
|
+
end
|
1411
|
+
list
|
1412
|
+
end
|
1413
|
+
|
1414
|
+
$exptt_table = Array.new(326)
|
1415
|
+
(0...326).each{|i| $exptt_table[i]=10**i}
|
1416
|
+
def exptt(_B, k)
|
1417
|
+
if _B==10 && k>=0 && k<326
|
1418
|
+
$exptt_table[k]
|
1419
|
+
else
|
1420
|
+
_B**k
|
1421
|
+
end
|
1422
|
+
end
|
1423
|
+
|
1424
|
+
$logB_table = Array.new(37)
|
1425
|
+
(2...37).each{|b| $logB_table[b]=1.0/Math.log(b)}
|
1426
|
+
def logB(_B, x)
|
1427
|
+
if _B>=2 && _B<37
|
1428
|
+
Math.log(x)*$logB_table[_B]
|
1429
|
+
else
|
1430
|
+
Math.log(x)/Math.log(_B)
|
1431
|
+
end
|
1432
|
+
end
|
1433
|
+
|
1434
|
+
def float_to_digits_max(v,f,e,round_mode,min_e,p,b,_B)
|
1435
|
+
|
1436
|
+
case round_mode
|
1437
|
+
when :even
|
1438
|
+
roundl = roundh = f.even?
|
1439
|
+
when :inf
|
1440
|
+
roundl = true
|
1441
|
+
roundh = false
|
1442
|
+
when :zero
|
1443
|
+
roundl = false
|
1444
|
+
roundh = true
|
1445
|
+
else
|
1446
|
+
# here we don't assume any rounding in the floating point numbers
|
1447
|
+
# the result is valid for any rounding
|
1448
|
+
roundl = false
|
1449
|
+
roundh = false
|
1450
|
+
end
|
1451
|
+
|
1452
|
+
if e >= 0
|
1453
|
+
if f != exptt(b,p-1)
|
1454
|
+
be = exptt(b,e)
|
1455
|
+
r,s,m_p,m_m,k = scale(f*be*2,2,be,be,0,_B,roundl ,roundh,v)
|
1456
|
+
else
|
1457
|
+
be = exptt(b,e)
|
1458
|
+
be1 = be*b
|
1459
|
+
r,s,m_p,m_m,k = scale(f*be1*2,b*2,be1,be,0,_B,roundl ,roundh,v)
|
1460
|
+
end
|
1461
|
+
else
|
1462
|
+
if e==min_e or f != exptt(b,p-1)
|
1463
|
+
r,s,m_p,m_m,k = scale(f*2,exptt(b,-e)*2,1,1,0,_B,roundl ,roundh,v)
|
1464
|
+
else
|
1465
|
+
r,s,m_p,m_m,k = scale(f*b*2,exptt(b,1-e)*2,b,1,0,_B,roundl ,roundh,v)
|
1466
|
+
end
|
1467
|
+
end
|
1468
|
+
[k]+generate_max(r,s,m_p,m_m,_B,roundl ,roundh)
|
1469
|
+
end
|
1470
|
+
|
1471
|
+
def generate_max(r,s,m_p,m_m,_B,low_ok ,high_ok)
|
1472
|
+
list = [false]
|
1473
|
+
loop do
|
1474
|
+
d,r = (r*_B).divmod(s)
|
1475
|
+
m_p *= _B
|
1476
|
+
m_m *= _B
|
1477
|
+
|
1478
|
+
list << d
|
1479
|
+
|
1480
|
+
tc1 = low_ok ? (r<=m_m) : (r<m_m)
|
1481
|
+
tc2 = high_ok ? (r+m_p >= s) : (r+m_p > s)
|
1482
|
+
|
1483
|
+
if tc1 && tc2
|
1484
|
+
list[0] = true if r*2 >= s
|
1485
|
+
break
|
1486
|
+
end
|
1487
|
+
end
|
1488
|
+
list
|
1489
|
+
end
|
1490
|
+
|
1491
|
+
end
|
1492
|
+
|
1493
|
+
end
|
1494
|
+
|
1495
|
+
class Float
|
1496
|
+
include Nio::Formattable
|
1497
|
+
def self.nio_read_neutral(neutral)
|
1498
|
+
x = nil
|
1499
|
+
|
1500
|
+
honor_rounding = true
|
1501
|
+
|
1502
|
+
if neutral.special?
|
1503
|
+
case neutral.special
|
1504
|
+
when :nan
|
1505
|
+
x = 0.0/0.0
|
1506
|
+
when :inf
|
1507
|
+
x = (neutral.sign=='-' ? -1.0 : +1.0)/0.0
|
1508
|
+
end
|
1509
|
+
elsif neutral.rep_pos<neutral.digits.length
|
1510
|
+
|
1511
|
+
x,y = neutral.to_RepDec.getQ
|
1512
|
+
x = Float(x)/y
|
1513
|
+
|
1514
|
+
else
|
1515
|
+
nd = neutral.base==10 ? Float::DIG : ((Float::MANT_DIG-1)*Math.log(2)/Math.log(neutral.base)).floor
|
1516
|
+
k = neutral.dec_pos-neutral.digits.length
|
1517
|
+
if !honor_rounding && (neutral.digits.length<=nd && k.abs<=15)
|
1518
|
+
x = neutral.digits.to_i(neutral.base).to_f
|
1519
|
+
if k<0
|
1520
|
+
x /= Float(neutral.base**-k)
|
1521
|
+
else
|
1522
|
+
x *= Float(neutral.base**k)
|
1523
|
+
end
|
1524
|
+
x = -x if neutral.sign=='-'
|
1525
|
+
elsif !honor_rounding && (k>0 && (k+neutral.digits.length < 2*nd))
|
1526
|
+
j = k-neutral.digits.length
|
1527
|
+
x = neutral.digits.to_i(neutral.base).to_f * Float(neutral.base**(j))
|
1528
|
+
x *= Float(neutral.base**(k-j))
|
1529
|
+
x = -x if neutral.sign=='-'
|
1530
|
+
elsif neutral.base.modulo(Float::RADIX)==0
|
1531
|
+
|
1532
|
+
f = neutral.digits.to_i(neutral.base)
|
1533
|
+
e = neutral.dec_pos-neutral.digits.length
|
1534
|
+
|
1535
|
+
rounding = neutral.rounding
|
1536
|
+
|
1537
|
+
x = Nio::Clinger::algM(f,e,rounding,neutral.base,Float::RADIX,Float::MANT_DIG,Float::MIN_EXP-Float::MANT_DIG,Float::MAX_EXP-Float::MANT_DIG)
|
1538
|
+
x = -x if neutral.sign=='-'
|
1539
|
+
|
1540
|
+
else
|
1541
|
+
|
1542
|
+
f = neutral.digits.to_i(neutral.base)
|
1543
|
+
e = neutral.dec_pos-neutral.digits.length
|
1544
|
+
|
1545
|
+
rounding = neutral.rounding
|
1546
|
+
|
1547
|
+
x = Nio::Clinger::algM(f,e,rounding,neutral.base,Float::RADIX,Float::MANT_DIG,Float::MIN_EXP-Float::MANT_DIG,Float::MAX_EXP-Float::MANT_DIG)
|
1548
|
+
x = -x if neutral.sign=='-'
|
1549
|
+
|
1550
|
+
end
|
1551
|
+
end
|
1552
|
+
|
1553
|
+
return x
|
1554
|
+
end
|
1555
|
+
def nio_write_neutral(fmt)
|
1556
|
+
neutral = Nio::NeutralNum.new
|
1557
|
+
x = self
|
1558
|
+
|
1559
|
+
if x.nan?
|
1560
|
+
neutral.set_special(:nan)
|
1561
|
+
elsif x.infinite?
|
1562
|
+
neutral.set_special(:inf, x<0 ? '-' : '+')
|
1563
|
+
else
|
1564
|
+
converted = false
|
1565
|
+
if fmt.get_ndig==:exact && fmt.get_approx==:simplify
|
1566
|
+
|
1567
|
+
if x!=0
|
1568
|
+
q = x.nio_r(Nio::Tolerance.decimals(Float::DIG,:sig))
|
1569
|
+
if q!=0
|
1570
|
+
neutral = q.nio_write_neutral(fmt)
|
1571
|
+
converted = true if neutral.digits.length<=Float::DIG
|
1572
|
+
end
|
1573
|
+
end
|
1574
|
+
|
1575
|
+
elsif fmt.get_approx==:exact
|
1576
|
+
neutral = x.nio_xr.nio_write_neutral(fmt)
|
1577
|
+
converted = true
|
1578
|
+
end
|
1579
|
+
if !converted
|
1580
|
+
if fmt.get_base==10 && false
|
1581
|
+
txt = format "%.*e",Float::DECIMAL_DIG-1,x # note that spec. e output precision+1 significant digits
|
1582
|
+
|
1583
|
+
sign = '+'
|
1584
|
+
if txt[0,1]=='-'
|
1585
|
+
sign = '-'
|
1586
|
+
txt = txt[1...txt.length]
|
1587
|
+
end
|
1588
|
+
exp = 0
|
1589
|
+
x_char = fmt.get_exp_char(fmt.get_base)
|
1590
|
+
|
1591
|
+
exp_i = txt.index(x_char)
|
1592
|
+
exp_i = txt.index(x_char.downcase) if exp_i===nil
|
1593
|
+
if exp_i!=nil
|
1594
|
+
exp = txt[exp_i+1...txt.length].to_i
|
1595
|
+
txt = txt[0...exp_i]
|
1596
|
+
end
|
1597
|
+
|
1598
|
+
dec_pos = txt.index '.'
|
1599
|
+
if dec_pos==nil
|
1600
|
+
dec_pos = txt.length
|
1601
|
+
else
|
1602
|
+
txt[dec_pos]=''
|
1603
|
+
end
|
1604
|
+
dec_pos += exp
|
1605
|
+
neutral.set sign, txt, dec_pos, nil, fmt.get_base_digits(10), true, fmt.get_round
|
1606
|
+
|
1607
|
+
converted = true
|
1608
|
+
end
|
1609
|
+
end
|
1610
|
+
if !converted
|
1611
|
+
|
1612
|
+
sign = x<0 ? '-' : '+'
|
1613
|
+
x = -x if sign=='-'
|
1614
|
+
f,e = Math.frexp(x)
|
1615
|
+
if e < Float::MIN_EXP
|
1616
|
+
# denormalized number
|
1617
|
+
f = Math.ldexp(f,e-Float::MIN_EXP+Float::MANT_DIG)
|
1618
|
+
e = Float::MIN_EXP-Float::MANT_DIG
|
1619
|
+
else
|
1620
|
+
# normalized number
|
1621
|
+
f = Math.ldexp(f,Float::MANT_DIG)
|
1622
|
+
e -= Float::MANT_DIG
|
1623
|
+
end
|
1624
|
+
f = f.to_i
|
1625
|
+
inexact = true
|
1626
|
+
|
1627
|
+
rounding = fmt.get_round
|
1628
|
+
|
1629
|
+
if fmt.get_all_digits?
|
1630
|
+
# use as many digits as possible
|
1631
|
+
dec_pos,r,*digits = Nio::BurgerDybvig::float_to_digits_max(x,f,e,rounding,Float::MIN_EXP-Float::MANT_DIG,Float::MANT_DIG,Float::RADIX,fmt.get_base)
|
1632
|
+
inexact = :roundup if r
|
1633
|
+
else
|
1634
|
+
# use as few digits as possible
|
1635
|
+
dec_pos,*digits = Nio::BurgerDybvig::float_to_digits(x,f,e,rounding,Float::MIN_EXP-Float::MANT_DIG,Float::MANT_DIG,Float::RADIX,fmt.get_base)
|
1636
|
+
end
|
1637
|
+
txt = ''
|
1638
|
+
digits.each{|d| txt << fmt.get_base_digits.digit_char(d)}
|
1639
|
+
neutral.set sign, txt, dec_pos, nil, fmt.get_base_digits, inexact, fmt.get_round
|
1640
|
+
|
1641
|
+
end
|
1642
|
+
end
|
1643
|
+
|
1644
|
+
return neutral
|
1645
|
+
end
|
1646
|
+
end
|
1647
|
+
|
1648
|
+
class Numeric
|
1649
|
+
unless method_defined?(:even?)
|
1650
|
+
def even?
|
1651
|
+
self.modulo(2)==0
|
1652
|
+
end
|
1653
|
+
end
|
1654
|
+
unless method_defined?(:odd?)
|
1655
|
+
def odd?
|
1656
|
+
self.modulo(2)!=0
|
1657
|
+
end
|
1658
|
+
end
|
1659
|
+
end
|
1660
|
+
|
1661
|
+
class Integer
|
1662
|
+
include Nio::Formattable
|
1663
|
+
def self.nio_read_neutral(neutral)
|
1664
|
+
x = nil
|
1665
|
+
|
1666
|
+
if neutral.special?
|
1667
|
+
raise Nio::InvalidFormat,"Invalid integer numeral"
|
1668
|
+
elsif neutral.rep_pos<neutral.digits.length
|
1669
|
+
return Rational.nio_read_neutral(neutral).to_i
|
1670
|
+
else
|
1671
|
+
digits = neutral.digits
|
1672
|
+
|
1673
|
+
if neutral.dec_pos <= 0
|
1674
|
+
digits = '0'
|
1675
|
+
elsif neutral.dec_pos <= digits.length
|
1676
|
+
digits = digits[0...neutral.dec_pos]
|
1677
|
+
else
|
1678
|
+
digits = digits + '0'*(neutral.dec_pos-digits.length)
|
1679
|
+
end
|
1680
|
+
|
1681
|
+
x = digits.to_i(neutral.base)
|
1682
|
+
# this was formely needed because we didn't adust the digits
|
1683
|
+
# if neutral.dec_pos != neutral.digits.length
|
1684
|
+
# # with rational included, negative powers of ten are rational numbers
|
1685
|
+
# x = (x*((neutral.base)**(neutral.dec_pos-neutral.digits.length))).to_i
|
1686
|
+
# end
|
1687
|
+
x = -x if neutral.sign=='-'
|
1688
|
+
end
|
1689
|
+
|
1690
|
+
return x
|
1691
|
+
end
|
1692
|
+
def nio_write_neutral(fmt)
|
1693
|
+
neutral = Nio::NeutralNum.new
|
1694
|
+
x = self
|
1695
|
+
|
1696
|
+
sign = x<0 ? '-' : '+'
|
1697
|
+
txt = x.abs.to_s(fmt.get_base)
|
1698
|
+
dec_pos = rep_pos = txt.length
|
1699
|
+
neutral.set sign, txt, dec_pos, nil, fmt.get_base_digits, false ,fmt.get_round
|
1700
|
+
|
1701
|
+
return neutral
|
1702
|
+
end
|
1703
|
+
end
|
1704
|
+
|
1705
|
+
class Rational
|
1706
|
+
include Nio::Formattable
|
1707
|
+
def self.nio_read_neutral(neutral)
|
1708
|
+
x = nil
|
1709
|
+
|
1710
|
+
if neutral.special?
|
1711
|
+
case neutral.special
|
1712
|
+
when :nan
|
1713
|
+
x = Rational(0,0)
|
1714
|
+
when :inf
|
1715
|
+
x = Rational((neutral.sign=='-' ? -1 : +1),0)
|
1716
|
+
end
|
1717
|
+
else
|
1718
|
+
x = Rational(*neutral.to_RepDec.getQ)
|
1719
|
+
end
|
1720
|
+
|
1721
|
+
return x
|
1722
|
+
end
|
1723
|
+
def nio_write_neutral(fmt)
|
1724
|
+
neutral = Nio::NeutralNum.new
|
1725
|
+
x = self
|
1726
|
+
|
1727
|
+
if x.denominator==0
|
1728
|
+
if x.numerator>0
|
1729
|
+
neutral.set_special(:inf)
|
1730
|
+
elsif x.numerator<0
|
1731
|
+
neutral.set_special(:inf,'-')
|
1732
|
+
else
|
1733
|
+
neutral.set_special(:nan)
|
1734
|
+
end
|
1735
|
+
else
|
1736
|
+
if fmt.get_base==10
|
1737
|
+
rd = Nio::RepDec.new.setQ(x.numerator,x.denominator)
|
1738
|
+
else
|
1739
|
+
opt = Nio::RepDec::DEF_OPT.dup.set_digits(fmt.get_base_digits)
|
1740
|
+
rd = Nio::RepDec.new.setQ(x.numerator,x.denominator, opt)
|
1741
|
+
end
|
1742
|
+
neutral = rd.to_NeutralNum(fmt.get_base_digits)
|
1743
|
+
neutral.rounding = fmt.get_round
|
1744
|
+
end
|
1745
|
+
|
1746
|
+
return neutral
|
1747
|
+
end
|
1748
|
+
end
|
1749
|
+
|
1750
|
+
if defined? BigDecimal
|
1751
|
+
class BigDecimal
|
1752
|
+
include Nio::Formattable
|
1753
|
+
def self.nio_read_neutral(neutral)
|
1754
|
+
x = nil
|
1755
|
+
|
1756
|
+
if neutral.special?
|
1757
|
+
case neutral.special
|
1758
|
+
when :nan
|
1759
|
+
x = BigDecimal('NaN') # BigDecimal("0")/0
|
1760
|
+
when :inf
|
1761
|
+
x = BigDecimal(neutral.sign=='-' ? '-1.0' : '+1.0')/0
|
1762
|
+
end
|
1763
|
+
elsif neutral.rep_pos<neutral.digits.length
|
1764
|
+
|
1765
|
+
x,y = neutral.to_RepDec.getQ
|
1766
|
+
x = BigDecimal(x.to_s)/y
|
1767
|
+
|
1768
|
+
else
|
1769
|
+
if neutral.base==10
|
1770
|
+
#x = BigDecimal(neutral.digits)
|
1771
|
+
#x *= BigDecimal("1E#{(neutral.dec_pos-neutral.digits.length)}")
|
1772
|
+
#x = -x if neutral.sign=='-'
|
1773
|
+
str = neutral.sign
|
1774
|
+
str += neutral.digits
|
1775
|
+
str += "E#{(neutral.dec_pos-neutral.digits.length)}"
|
1776
|
+
x = BigDecimal(str)
|
1777
|
+
else
|
1778
|
+
x = BigDecimal(neutral.digits.to_i(neutral.base).to_s)
|
1779
|
+
x *= BigDecimal(neutral.base.to_s)**(neutral.dec_pos-neutral.digits.length)
|
1780
|
+
x = -x if neutral.sign=='-'
|
1781
|
+
end
|
1782
|
+
end
|
1783
|
+
|
1784
|
+
return x
|
1785
|
+
end
|
1786
|
+
def nio_write_neutral(fmt)
|
1787
|
+
neutral = Nio::NeutralNum.new
|
1788
|
+
x = self
|
1789
|
+
|
1790
|
+
if x.nan?
|
1791
|
+
neutral.set_special(:nan)
|
1792
|
+
elsif x.infinite?
|
1793
|
+
neutral.set_special(:inf, x<0 ? '-' : '+')
|
1794
|
+
else
|
1795
|
+
converted = false
|
1796
|
+
if fmt.get_ndig==:exact && fmt.get_approx==:simplify
|
1797
|
+
|
1798
|
+
prc = [x.precs[0],20].max
|
1799
|
+
neutral = x.nio_r(Nio::BigTolerance.decimals(prc,:sig)).nio_write_neutral(fmt)
|
1800
|
+
converted = true if neutral.digits.length<prc
|
1801
|
+
|
1802
|
+
elsif fmt.get_approx==:exact && fmt.get_base!=10
|
1803
|
+
neutral = x.nio_xr.nio_write_neutral(fmt)
|
1804
|
+
converted = true
|
1805
|
+
end
|
1806
|
+
if !converted
|
1807
|
+
if fmt.get_base==10
|
1808
|
+
txt = x.to_s
|
1809
|
+
|
1810
|
+
sign = '+'
|
1811
|
+
if txt[0,1]=='-'
|
1812
|
+
sign = '-'
|
1813
|
+
txt = txt[1...txt.length]
|
1814
|
+
end
|
1815
|
+
exp = 0
|
1816
|
+
x_char = fmt.get_exp_char(fmt.get_base)
|
1817
|
+
|
1818
|
+
exp_i = txt.index(x_char)
|
1819
|
+
exp_i = txt.index(x_char.downcase) if exp_i===nil
|
1820
|
+
if exp_i!=nil
|
1821
|
+
exp = txt[exp_i+1...txt.length].to_i
|
1822
|
+
txt = txt[0...exp_i]
|
1823
|
+
end
|
1824
|
+
|
1825
|
+
dec_pos = txt.index '.'
|
1826
|
+
if dec_pos==nil
|
1827
|
+
dec_pos = txt.length
|
1828
|
+
else
|
1829
|
+
txt[dec_pos]=''
|
1830
|
+
end
|
1831
|
+
dec_pos += exp
|
1832
|
+
neutral.set sign, txt, dec_pos, nil, fmt.get_base_digits(10), true, fmt.get_round
|
1833
|
+
|
1834
|
+
end
|
1835
|
+
end
|
1836
|
+
if !converted
|
1837
|
+
|
1838
|
+
min_prec = 24
|
1839
|
+
min_exp = -1000
|
1840
|
+
s,f,b,e = x.split
|
1841
|
+
e -= f.size
|
1842
|
+
sign = s<0 ? '-' : '+'
|
1843
|
+
x = -x if sign=='-'
|
1844
|
+
f_i = f.to_i
|
1845
|
+
prc = [x.precs[0],min_prec].max
|
1846
|
+
f_i *= 10**(prc-f.size)
|
1847
|
+
e -= (prc-f.size)
|
1848
|
+
|
1849
|
+
inexact = true
|
1850
|
+
|
1851
|
+
rounding = fmt.get_round
|
1852
|
+
|
1853
|
+
if fmt.get_all_digits?
|
1854
|
+
# use as many digits as possible
|
1855
|
+
dec_pos,r,*digits = Nio::BurgerDybvig::float_to_digits_max(x,f_i,e,rounding,[e,min_exp].min,prc,b,fmt.get_base)
|
1856
|
+
inexact = :roundup if r
|
1857
|
+
else
|
1858
|
+
# use as few digits as possible
|
1859
|
+
dec_pos,*digits = Nio::BurgerDybvig::float_to_digits(x,f_i,e,rounding,[e,min_exp].min,prc,b,fmt.get_base)
|
1860
|
+
end
|
1861
|
+
txt = ''
|
1862
|
+
digits.each{|d| txt << fmt.get_base_digits.digit_char(d)}
|
1863
|
+
neutral.set sign, txt, dec_pos, nil, fmt.get_base_digits, inexact, fmt.get_round
|
1864
|
+
|
1865
|
+
end
|
1866
|
+
end
|
1867
|
+
|
1868
|
+
return neutral
|
1869
|
+
end
|
1870
|
+
end
|
1871
|
+
end
|
1872
|
+
|