Skip to content

Commit 01fe7c2

Browse files
ararslanStefanKarpinski
authored andcommitted
Allow passing a function to extrema (#30323)
Currently `minimum` and `maximum` can accept a function argument, but `extrema` cannot. This makes it consistent.
1 parent 43c6b57 commit 01fe7c2

File tree

4 files changed

+65
-21
lines changed

4 files changed

+65
-21
lines changed

NEWS.md

+3
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,8 @@ Julia v1.2 Release Notes
44
New language features
55
---------------------
66

7+
* The `extrema` function now accepts a function argument in the same manner as `minimum` and
8+
`maximum` ([#30323]).
79

810
Language changes
911
----------------
@@ -36,3 +38,4 @@ Deprecated or removed
3638

3739
<!--- generated by NEWS-update.jl: -->
3840
[#29998]: https://github.com/JuliaLang/julia/issues/29998
41+
[#30323]: https://github.com/JuliaLang/julia/issues/30323

base/multidimensional.jl

+26-13
Original file line numberDiff line numberDiff line change
@@ -1500,40 +1500,53 @@ julia> extrema(A, dims = (1,2))
15001500
(9, 15)
15011501
```
15021502
"""
1503-
extrema(A::AbstractArray; dims = :) = _extrema_dims(A, dims)
1503+
extrema(A::AbstractArray; dims = :) = _extrema_dims(identity, A, dims)
15041504

1505-
_extrema_dims(A::AbstractArray, ::Colon) = _extrema_itr(A)
1505+
"""
1506+
extrema(f, A::AbstractArray; dims) -> Array{Tuple}
1507+
1508+
Compute the minimum and maximum of `f` applied to each element in the given dimensions
1509+
of `A`.
1510+
1511+
!!! compat "Julia 1.2"
1512+
This method requires Julia 1.2 or later.
1513+
"""
1514+
extrema(f, A::AbstractArray; dims=:) = _extrema_dims(f, A, dims)
1515+
1516+
_extrema_dims(f, A::AbstractArray, ::Colon) = _extrema_itr(f, A)
15061517

1507-
function _extrema_dims(A::AbstractArray, dims)
1518+
function _extrema_dims(f, A::AbstractArray, dims)
15081519
sz = [size(A)...]
15091520
for d in dims
15101521
sz[d] = 1
15111522
end
1512-
B = Array{Tuple{eltype(A),eltype(A)}}(undef, sz...)
1513-
return extrema!(B, A)
1523+
T = promote_op(f, eltype(A))
1524+
B = Array{Tuple{T,T}}(undef, sz...)
1525+
return extrema!(f, B, A)
15141526
end
15151527

1516-
@noinline function extrema!(B, A)
1528+
@noinline function extrema!(f, B, A)
15171529
@assert !has_offset_axes(B, A)
15181530
sA = size(A)
15191531
sB = size(B)
15201532
for I in CartesianIndices(sB)
1521-
AI = A[I]
1522-
B[I] = (AI, AI)
1533+
fAI = f(A[I])
1534+
B[I] = (fAI, fAI)
15231535
end
15241536
Bmax = CartesianIndex(sB)
15251537
@inbounds @simd for I in CartesianIndices(sA)
15261538
J = min(Bmax,I)
15271539
BJ = B[J]
1528-
AI = A[I]
1529-
if AI < BJ[1]
1530-
B[J] = (AI, BJ[2])
1531-
elseif AI > BJ[2]
1532-
B[J] = (BJ[1], AI)
1540+
fAI = f(A[I])
1541+
if fAI < BJ[1]
1542+
B[J] = (fAI, BJ[2])
1543+
elseif fAI > BJ[2]
1544+
B[J] = (BJ[1], fAI)
15331545
end
15341546
end
15351547
return B
15361548
end
1549+
extrema!(B, A) = extrema!(identity, B, A)
15371550

15381551
# Show for pairs() with Cartesian indices. Needs to be here rather than show.jl for bootstrap order
15391552
function Base.showarg(io::IO, r::Iterators.Pairs{<:Integer, <:Any, <:Any, T}, toplevel) where T <: Union{AbstractVector, Tuple}

base/operators.jl

+24-5
Original file line numberDiff line numberDiff line change
@@ -440,24 +440,43 @@ julia> extrema([9,pi,4.5])
440440
(3.141592653589793, 9.0)
441441
```
442442
"""
443-
extrema(itr) = _extrema_itr(itr)
443+
extrema(itr) = _extrema_itr(identity, itr)
444444

445-
function _extrema_itr(itr)
445+
"""
446+
extrema(f, itr) -> Tuple
447+
448+
Compute both the minimum and maximum of `f` applied to each element in `itr` and return
449+
them as a 2-tuple. Only one pass is made over `itr`.
450+
451+
!!! compat "Julia 1.2"
452+
This method requires Julia 1.2 or later.
453+
454+
# Examples
455+
```jldoctest
456+
julia> extrema(sin, 0:π)
457+
(0.0, 0.9092974268256817)
458+
```
459+
"""
460+
extrema(f, itr) = _extrema_itr(f, itr)
461+
462+
function _extrema_itr(f, itr)
446463
y = iterate(itr)
447464
y === nothing && throw(ArgumentError("collection must be non-empty"))
448465
(v, s) = y
449-
vmin = vmax = v
466+
vmin = vmax = f(v)
450467
while true
451468
y = iterate(itr, s)
452469
y === nothing && break
453470
(x, s) = y
454-
vmax = max(x, vmax)
455-
vmin = min(x, vmin)
471+
fx = f(x)
472+
vmax = max(fx, vmax)
473+
vmin = min(fx, vmin)
456474
end
457475
return (vmin, vmax)
458476
end
459477

460478
extrema(x::Real) = (x, x)
479+
extrema(f, x::Real) = (y = f(x); (y, y))
461480

462481
## definitions providing basic traits of arithmetic operators ##
463482

test/reduce.jl

+12-3
Original file line numberDiff line numberDiff line change
@@ -180,10 +180,17 @@ prod2(itr) = invoke(prod, Tuple{Any}, itr)
180180
@test maximum(5) == 5
181181
@test minimum(5) == 5
182182
@test extrema(5) == (5, 5)
183+
@test extrema(abs2, 5) == (25, 25)
183184

184-
@test maximum([4, 3, 5, 2]) == 5
185-
@test minimum([4, 3, 5, 2]) == 2
186-
@test extrema([4, 3, 5, 2]) == (2, 5)
185+
let x = [4,3,5,2]
186+
@test maximum(x) == 5
187+
@test minimum(x) == 2
188+
@test extrema(x) == (2, 5)
189+
190+
@test maximum(abs2, x) == 25
191+
@test minimum(abs2, x) == 4
192+
@test extrema(abs2, x) == (4, 25)
193+
end
187194

188195
@test isnan(maximum([NaN]))
189196
@test isnan(minimum([NaN]))
@@ -211,6 +218,7 @@ prod2(itr) = invoke(prod, Tuple{Any}, itr)
211218

212219
@test maximum(abs2, 3:7) == 49
213220
@test minimum(abs2, 3:7) == 9
221+
@test extrema(abs2, 3:7) == (9, 49)
214222

215223
@test maximum(Int16[1]) === Int16(1)
216224
@test maximum(Vector(Int16(1):Int16(100))) === Int16(100)
@@ -227,6 +235,7 @@ A = circshift(reshape(1:24,2,3,4), (0,1,1))
227235
@test size(extrema(A,dims=1)) == size(maximum(A,dims=1))
228236
@test size(extrema(A,dims=(1,2))) == size(maximum(A,dims=(1,2)))
229237
@test size(extrema(A,dims=(1,2,3))) == size(maximum(A,dims=(1,2,3)))
238+
@test extrema(x->div(x, 2), A, dims=(2,3)) == reshape([(0,11),(1,12)],2,1,1)
230239

231240
# any & all
232241

0 commit comments

Comments
 (0)