Skip to content

HTTPS clone URL

Subversion checkout URL

You can clone with
or
.
Download ZIP
Fetching contributors…

Cannot retrieve contributors at this time

60 lines (50 sloc) 1.455 kb
# -*- encoding: us-ascii -*-
# Float#rationalize and String#to_r require complex algorithms. The Regexp
# required for String#to_r cannot be created at class scope when loading the
# kernel because Regexp needs to load after String and making a simpler
# Regexp#initialize for bootstrapping isn't really feasible. So these
# algorithms are located here.
#
class String
class Rationalizer
SPACE = "\\s*"
DIGITS = "(?:[0-9](?:_[0-9]|[0-9])*)"
NUMERATOR = "(?:#{DIGITS}?\\.)?#{DIGITS}(?:[eE][-+]?#{DIGITS})?"
DENOMINATOR = DIGITS
RATIONAL = "\\A#{SPACE}([-+])?(#{NUMERATOR})(?:\\/(#{DENOMINATOR}))?#{SPACE}"
PATTERN = Regexp.new RATIONAL
def initialize(value)
@value = value
end
def convert
if m = PATTERN.match(@value)
si = m[1]
nu = m[2]
de = m[3]
re = m.post_match
ifp, exp = nu.split /[eE]/
ip, fp = ifp.split /\./
value = Rational(ip.to_i)
if fp
ctype = Rubinius::CType
i = count = 0
size = fp.size
while i < size
count += 1 if ctype.isdigit fp.getbyte(i)
i += 1
end
l = 10 ** count
value *= l
value += fp.to_i
value /= l
end
value = -value if si == "-"
value *= 10 ** exp.to_i if exp
value /= de.to_i if de
value
else
Rational(0, 1)
end
end
end
end
Jump to Line
Something went wrong with that request. Please try again.