[ANN] Announcing RangeHelpers.jl

Ever needed a range with startpoint 10, endpoint 121.7 and a step of 25?
Well that is mathematically not possible, so you need to compromise.
There are lots of options, you could relax the startpoint, endpoint or step. In the past doing this was annoying and prone to off-by-one-errors:

julia> Base.range(10, step=25, length=round(Int, (121.7-10)/25)); # is it correct??

RangeHelpers.jl aims to solve range construction headaches once and for all:

julia> using RangeHelpers: range

julia> using RangeHelpers

julia> range(start=10, stop=121.7, step=around(25)) # compromise on step
10.0:27.925:121.7

julia> range(start=10, stop=121.7, step=below(25))  # compromise step at most 25
10.0:22.34:121.7

julia> range(start=10, stop=above(121.7), step=25)  # exact step, but allow bigger endpoint
10:25:135

julia> anchorrange(42, start=around(10), step=25, stop=around(121.7)) # make sure 42 is on the grid
17.0:25.0:117.0

See the documentation for even more ways to make ranges.

19 Likes

Nice concept!

One thing that came to my mind: Exporting functions with common names like above, around, etc. has a high conflict potential with other packages and local names. Just a suggestion - how would you like

range(start=10, stop=121.7, step=(≈,25))
range(start=10, stop=121.7, step=(<,25))

or so?

3 Likes

One thing that came to my mind: Exporting functions with common names like above , around , etc. has a high conflict potential with other packages and local names.

Absolutely! Personally I use import X as Y a lot:

import RangeHelpers as RH
RH.range(start=RH.below(4), step=2, stop=3)

The syntax you suggest is pretty clever, feel free to open a PR!

1 Like

Thanks!

Or make those modifiers types and then there is no conflict (and I bet some benefit to the speed?).

1 Like