Skip to content

Commit

Permalink
FloatRange: new type for accurate floating-point ranges [fix #2333]
Browse files Browse the repository at this point in the history
  • Loading branch information
StefanKarpinski committed Feb 18, 2014
1 parent 7d997ed commit 1467413
Show file tree
Hide file tree
Showing 3 changed files with 96 additions and 19 deletions.
71 changes: 56 additions & 15 deletions base/range.jl
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,15 @@ immutable Range1{T<:Real} <: Ranges{T}
end
Range1{T}(start::T, len::Integer) = Range1{T}(start, len)

immutable FloatRange{T<:FloatingPoint} <: Ranges{T}
start::T
step::T
divisor::T
len::T
end
FloatRange(a::FloatingPoint, s::FloatingPoint, d::FloatingPoint, l::Real) =
FloatRange{promote_type(typeof(a),typeof(s),typeof(d))}(a,s,d,l)

function colon{T<:Integer}(start::T, step::T, stop::T)
step != 0 || error("step cannot be zero in colon syntax")
Range{T}(start, step, max(0, 1 + fld(stop-start, step)))
Expand All @@ -59,7 +68,7 @@ end

function colon{T<:Real}(start::T, step::T, stop::T)
step != 0 || error("step cannot be zero in colon syntax")
if (step<0) != (stop<start)
if (step < 0) != (stop < start)
len = 0
else
nf = (stop-start)/step + 1
Expand Down Expand Up @@ -104,17 +113,46 @@ end
colon(start::Real, step::Real, stop::Real) = colon(promote(start, step, stop)...)
colon(start::Real, stop::Real) = colon(promote(start, stop)...)

function colon{T<:FloatingPoint}(start::T, step::T, stop::T)
step != 0 || error("step cannot be zero in colon syntax")
(step < 0) != (stop < start) && return FloatRange{T}(start,step,1,0)
ratio = (stop-start)/step
divisor = one(step)
rounded = round(ratio)
len = floor(ratio)+1
local num
den, t, x = zero(step), one(step), abs(step)
while true
m = trunc(x); x -= m
t, den = den, m*den + t
num = round(den*step)
(x == 0 || num/den == step) && break
x = inv(x)
end
if (start*den + rounded*num)/den == stop &&
(stop*den - start*den)/rounded/den == step
start *= den
step = num
divisor = den
len = rounded+1
end
FloatRange{T}(start,step,divisor,len)
end

similar(r::Ranges, T::Type, dims::Dims) = Array(T, dims)

length(r::Ranges) = r.len
size(r::Ranges) = (r.len,)
length(r::Ranges) = integer(r.len)
size(r::Ranges) = (length(r),)
isempty(r::Ranges) = r.len==0
first(r::Ranges) = r.start
first(r::FloatRange) = r.start/r.divisor
last{T}(r::Range1{T}) = oftype(T, r.start + r.len-1)
last{T}(r::Range{T}) = oftype(T, r.start + (r.len-1)*r.step)
last{T}(r::FloatRange{T}) = oftype(T, (r.start + (r.len-1)*r.step)/r.divisor)

step(r::Range) = r.step
step(r::Range) = r.step
step(r::Range1) = one(r.start)
step(r::FloatRange) = r.step/r.divisor

minimum(r::Range1) = isempty(r) ? error("range must be non-empty") : first(r)
maximum(r::Range1) = isempty(r) ? error("range must be non-empty") : last(r)
Expand All @@ -130,9 +168,13 @@ copy(r::Ranges) = r
getindex(r::Ranges, i::Real) = getindex(r, to_index(i))

function getindex{T}(r::Ranges{T}, i::Integer)
if !(1 <= i <= r.len); error(BoundsError); end
1 <= i <= r.len || error(BoundsError)
oftype(T, r.start + (i-1)*step(r))
end
function getindex{T}(r::FloatRange{T}, i::Integer)
1 <= i <= r.len || error(BoundsError)
oftype(T, (r.start + (i-1)*r.step)/r.divisor)
end

function getindex(r::Range1, s::Range1{Int})
if s.len > 0
Expand All @@ -156,18 +198,16 @@ function getindex(r::Ranges, s::Ranges{Int})
end
end

function show(io::IO, r::Range)
if step(r) == 0
print(io, "Range(",r.start,",",step(r),",",r.len,")")
else
print(io, repr(r.start),':',repr(step(r)),':',repr(last(r)))
end
function show(io::IO, r::Union(Range,FloatRange))
step(r) == 0 ? invoke(show,(IO,Any),io,r) :
print(io, repr(first(r)), ':', repr(step(r)), ':', repr(last(r)))
end
show(io::IO, r::Range1) = print(io, repr(r.start),':',repr(last(r)))
show(io::IO, r::Range1) = print(io, repr(first(r)), ':', repr(last(r)))

start(r::Ranges) = 0
next{T}(r::Range{T}, i) = (oftype(T, r.start + i*step(r)), i+1)
next{T}(r::Range{T}, i) = (oftype(T, r.start + i*step(r)), i+1)
next{T}(r::Range1{T}, i) = (oftype(T, r.start + i), i+1)
next{T}(r::FloatRange{T}, i) = (oftype(T, (r.start + i*r.step)/r.divisor), i+1)
done(r::Ranges, i) = (length(r) <= i)

# though these look very similar to the above, for some reason LLVM generates
Expand All @@ -176,7 +216,7 @@ start{T<:Integer}(r::Range1{T}) = r.start
next{T<:Integer}(r::Range1{T}, i) = (i, oftype(T, i+1))
done{T<:Integer}(r::Range1{T}, i) = i==oftype(T, r.start+r.len)

==(r::Ranges, s::Ranges) = (r.start==s.start) & (step(r)==step(s)) & (r.len==s.len)
==(r::Ranges, s::Ranges) = (first(r)==first(s)) & (step(r)==step(s)) & (length(r)==length(s))
==(r::Range1, s::Range1) = (r.start==s.start) & (r.len==s.len)

# TODO: isless?
Expand Down Expand Up @@ -389,7 +429,8 @@ function vcat{T}(rs::Ranges{T}...)
return a
end

reverse{T<:Real}(r::Ranges{T}) = Range(last(r), -step(r), r.len)
reverse(r::Ranges) = Range(last(r), -step(r), r.len)
reverse(r::FloatRange) = FloatRange(last(r), -r.step, r.divisor, r.len)

## sorting ##

Expand Down
8 changes: 4 additions & 4 deletions test/random.jl
Original file line number Diff line number Diff line change
Expand Up @@ -6,12 +6,12 @@
@test minimum([rand(int32(1):int32(7^7)) for i = 1:100000]) > 0
@test(typeof(rand(false:true)) == Bool)

for T in (Int8, Uint8, Int16, Uint16, Int32, Uint32, Int64, Uint64, Int128, Uint128, Char, BigInt,
Float16, Float32, Float64, Rational{Int})
r = rand(convert(T, 97):convert(T, 122))
for T in (Int8, Uint8, Int16, Uint16, Int32, Uint32, Int64, Uint64, Int128, Uint128,
Char, BigInt, Float32, Float64, Rational{Int})
r = rand(convert(T,97):convert(T,122))
@test typeof(r) == T
@test 97 <= r <= 122
r = rand(convert(T, 97):convert(T,2):convert(T, 122),2)[1]
r = rand(convert(T,97):convert(T,2):convert(T,122),2)[1]
@test typeof(r) == T
@test 97 <= r <= 122
@test mod(r,2)==1
Expand Down
36 changes: 36 additions & 0 deletions test/ranges.jl
Original file line number Diff line number Diff line change
Expand Up @@ -159,3 +159,39 @@ else
@test sum(int64(1:10^9)) == div(10^9 * (int64(10^9)+1), 2)
@test sum(int64(1:10^9-1)) == div(10^9 * (int64(10^9)-1), 2)
end

# tricky floating-point ranges

@test 0.1:0.1:0.3 == [1:3]./10
@test 0.0:0.1:0.3 == [0:3]./10
@test 0.3:-0.1:-0.1 == [3:-1:-1]./10
@test 0.1:-0.1:-0.3 == [1:-1:-3]./10
@test 0.0:0.1:1.0 == [0:10]./10
@test 0.0:-0.1:1.0 == []
@test 0.0:0.1:-1.0 == []
@test 0.0:-0.1:-1.0 == [0:-1:-10]./10
@test 1.0:1/49:27.0 == [49:1323]./49
@test 0.0:0.7:2.1 == [0:7:21]./10
@test 0.0:1.1:3.3 == [0:11:33]./10
@test 0.1:1.1:3.4 == [1:11:34]./10
@test 0.0:1.3:3.9 == [0:13:39]./10
@test 0.1:1.3:4.0 == [1:13:40]./10
@test 1.1:1.1:3.3 == [11:11:33]./10
@test 0.3:0.1:1.1 == [3:1:11]./10

@test 0.0:1.0:5.5 == [0:10:55]./10
@test 0.0:-1.0:0.5 == []
@test 0.0:1.0:0.5 == [0.0]

@test prevfloat(0.1):0.1:0.3 == [prevfloat(0.1), 0.2, 0.3]
@test nextfloat(0.1):0.1:0.3 == [nextfloat(0.1), 0.2]
@test prevfloat(0.0):0.1:0.3 == [prevfloat(0.0), 0.1, 0.2, 0.3]
@test nextfloat(0.0):0.1:0.3 == [nextfloat(0.0), 0.1, 0.2, 0.3]
@test 0.1:0.1:prevfloat(0.3) == [0.1, 0.2]
@test 0.1:0.1:nextfloat(0.3) == [0.1, 0.2, nextfloat(0.3)]
@test 0.0:0.1:prevfloat(0.3) == [0.0, 0.1, 0.2]
@test 0.0:0.1:nextfloat(0.3) == [0.0, 0.1, 0.2, nextfloat(0.3)]
@test 0.1:prevfloat(0.1):0.3 == [0.1, 0.2, 0.3]
@test 0.1:nextfloat(0.1):0.3 == [0.1, 0.2]
@test 0.0:prevfloat(0.1):0.3 == [0.0, prevfloat(0.1), prevfloat(0.2), 0.3]
@test 0.0:nextfloat(0.1):0.3 == [0.0, nextfloat(0.1), nextfloat(0.2)]

0 comments on commit 1467413

Please sign in to comment.