Skip to content

Commit 5ff9e3a

Browse files
authored
Add range(; stop) and range(; length) (#39241)
As a single keyword arg only (single positional arg not allowed still)
1 parent 3306a8b commit 5ff9e3a

File tree

3 files changed

+59
-3
lines changed

3 files changed

+59
-3
lines changed

NEWS.md

+1
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@ New library features
4141
Standard library changes
4242
------------------------
4343

44+
* `range` accepts either `stop` or `length` as a sole keyword argument ([#39241])
4445
* The `length` function on certain ranges of certain specific element types no longer checks for integer
4546
overflow in most cases. The new function `checked_length` is now available, which will try to use checked
4647
arithmetic to error if the result may be wrapping. Or use a package such as SaferIntegers.jl when

base/range.jl

+39-2
Original file line numberDiff line numberDiff line change
@@ -58,6 +58,9 @@ Valid invocations of range are:
5858
* Call `range` with any three of `start`, `step`, `stop`, `length`.
5959
* Call `range` with two of `start`, `stop`, `length`. In this case `step` will be assumed
6060
to be one. If both arguments are Integers, a [`UnitRange`](@ref) will be returned.
61+
* Call `range` with one of `stop` or `length`. `start` and `step` will be assumed to be one.
62+
63+
See Extended Help for additional details on the returned type.
6164
6265
# Examples
6366
```jldoctest
@@ -87,6 +90,15 @@ julia> range(stop=10, step=1, length=5)
8790
8891
julia> range(start=1, step=1, stop=10)
8992
1:1:10
93+
94+
julia> range(; length = 10)
95+
Base.OneTo(10)
96+
97+
julia> range(; stop = 6)
98+
Base.OneTo(6)
99+
100+
julia> range(; stop = 6.5)
101+
1.0:1.0:6.0
90102
```
91103
If `length` is not specified and `stop - start` is not an integer multiple of `step`, a range that ends before `stop` will be produced.
92104
```jldoctest
@@ -103,6 +115,23 @@ To avoid this induced overhead, see the [`LinRange`](@ref) constructor.
103115
!!! compat "Julia 1.7"
104116
The versions without keyword arguments and `start` as a keyword argument
105117
require at least Julia 1.7.
118+
119+
!!! compat "Julia 1.8"
120+
The versions with `stop` as a sole keyword argument,
121+
or `length` as a sole keyword argument require at least Julia 1.8.
122+
123+
124+
# Extended Help
125+
126+
`range` will produce a `Base.OneTo` when the arguments are Integers and
127+
* Only `length` is provided
128+
* Only `stop` is provided
129+
130+
`range` will produce a `UnitRange` when the arguments are Integers and
131+
* Only `start` and `stop` are provided
132+
* Only `length` and `stop` are provided
133+
134+
A `UnitRange` is not produced if `step` is provided even if specified as one.
106135
"""
107136
function range end
108137

@@ -115,8 +144,8 @@ range(;start=nothing, stop=nothing, length::Union{Integer, Nothing}=nothing, ste
115144
_range(start, step, stop, length)
116145

117146
_range(start::Nothing, step::Nothing, stop::Nothing, len::Nothing) = range_error(start, step, stop, len)
118-
_range(start::Nothing, step::Nothing, stop::Nothing, len::Any ) = range_error(start, step, stop, len)
119-
_range(start::Nothing, step::Nothing, stop::Any , len::Nothing) = range_error(start, step, stop, len)
147+
_range(start::Nothing, step::Nothing, stop::Nothing, len::Any ) = range_length(len)
148+
_range(start::Nothing, step::Nothing, stop::Any , len::Nothing) = range_stop(stop)
120149
_range(start::Nothing, step::Nothing, stop::Any , len::Any ) = range_stop_length(stop, len)
121150
_range(start::Nothing, step::Any , stop::Nothing, len::Nothing) = range_error(start, step, stop, len)
122151
_range(start::Nothing, step::Any , stop::Nothing, len::Any ) = range_error(start, step, stop, len)
@@ -131,6 +160,14 @@ _range(start::Any , step::Any , stop::Nothing, len::Any ) = range_start
131160
_range(start::Any , step::Any , stop::Any , len::Nothing) = range_start_step_stop(start, step, stop)
132161
_range(start::Any , step::Any , stop::Any , len::Any ) = range_error(start, step, stop, len)
133162

163+
# Length as the only argument
164+
range_length(len::Integer) = OneTo(len)
165+
166+
# Stop as the only argument
167+
range_stop(stop) = range_start_stop(oneunit(stop), stop)
168+
range_stop(stop::Integer) = range_length(stop)
169+
170+
# Stop and length as the only argument
134171
range_stop_length(a::Real, len::Integer) = UnitRange{typeof(a)}(oftype(a, a-len+1), a)
135172
range_stop_length(a::AbstractFloat, len::Integer) = range_step_stop_length(oftype(a, 1), a, len)
136173
range_stop_length(a, len::Integer) = range_step_stop_length(oftype(a-a, 1), a, len)

test/ranges.jl

+19-1
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,20 @@ using Base.Checked: checked_length
2020
# the next ones use ==, because it changes the eltype
2121
@test r == range(first(r), last(r), length(r) )
2222
@test r == range(start=first(r), stop=last(r), length=length(r))
23+
@test r === range( stop=last(r), length=length(r))
24+
25+
r = 1:5
26+
o = Base.OneTo(5)
27+
let start=first(r), step=step(r), stop=last(r), length=length(r)
28+
@test o === range(; stop )
29+
@test o === range(; length)
30+
@test r === range(; start, stop )
31+
@test r === range(; stop, length)
32+
# the next three lines uses ==, because it changes the eltype
33+
@test r == range(; start, stop, length)
34+
@test r == range(; start, step, length)
35+
@test r == range(; stop=Float64(stop))
36+
end
2337

2438
for T = (Int8, Rational{Int16}, UInt32, Float64, Char)
2539
@test typeof(range(start=T(5), length=3)) === typeof(range(stop=T(5), length=3))
@@ -1508,8 +1522,12 @@ end
15081522
@test_throws ArgumentError range(1)
15091523
@test_throws ArgumentError range(nothing)
15101524
@test_throws ArgumentError range(1, step=4)
1511-
@test_throws ArgumentError range(nothing, length=2)
1525+
@test_throws ArgumentError range(; step=1, length=6)
1526+
@test_throws ArgumentError range(; step=2, stop=7.5)
15121527
@test_throws ArgumentError range(1.0, step=0.25, stop=2.0, length=5)
1528+
@test_throws ArgumentError range(; stop=nothing)
1529+
@test_throws ArgumentError range(; length=nothing)
1530+
@test_throws TypeError range(; length=5.5)
15131531
end
15141532

15151533
@testset "issue #23300#issuecomment-371575548" begin

0 commit comments

Comments
 (0)