From 4f14166f72b521bae65b25ad5a235d2d20733f95 Mon Sep 17 00:00:00 2001 From: Andrew Radcliffe Date: Wed, 22 Jun 2022 13:52:08 -0700 Subject: [PATCH] `findextrema`: compute `findmin` and `findmax` in single pass This is a simple extension of extant `findmin` and `findmax` methods. Depending on context (cost of `f`; whether reduction is over dims; size of array) the speedup increase is somewhere between 1.0-1.6 (no regressions). Interestingly, I noticed but could not locate a `findextrema`; there is some [mention](https://github.com/JuliaLang/julia/pull/7327) of it, but nothing in Base. If it was deemed unworthy, please excuse this errant PR. --- base/exports.jl | 2 + base/reduce.jl | 51 +++++++++++++ base/reducedim.jl | 147 ++++++++++++++++++++++++++++++++++++ doc/src/base/collections.md | 2 + test/reduce.jl | 11 +++ test/reducedim.jl | 81 ++++++++++++++++++++ 6 files changed, 294 insertions(+) diff --git a/base/exports.jl b/base/exports.jl index 266a4aa8038fb..0e513cf50f1f5 100644 --- a/base/exports.jl +++ b/base/exports.jl @@ -468,6 +468,8 @@ export findmin, findmin!, findmax!, + findextrema, + findextrema! findnext, findprev, match, diff --git a/base/reduce.jl b/base/reduce.jl index 0bcf5f8ca5923..82a5fd8e7ea6b 100644 --- a/base/reduce.jl +++ b/base/reduce.jl @@ -988,6 +988,57 @@ julia> findmin([1, 7, 7, NaN]) findmin(itr) = _findmin(itr, :) _findmin(a, ::Colon) = findmin(identity, a) +""" + findextrema(f, domain) -> ((f(x), index_mn), (f(x), index_mx)) + +Return the pair of pairs which would be returned by `(findmin(f, domain), findmax(f, domain))`, +but computed in a single pass. + +# Examples + +```jldoctest +julia> findextrema(identity, 5:9) +((5, 1), (9, 5)) + +julia> findextrema(-, 1:10) +((-10, 10), (-1, 1)) + +julia> findextrema(first, [(2, :a), (2, :b), (3, :c)]) +((2, 1), (3, 3)) + +julia> findextrema(cos, 0:π/2:2π) +((-1.0, 3), (1.0, 1)) +``` +""" +findextrema(f, domain) = _findextrema(f, domain, :) +_findextrema(f, domain, ::Colon) = mapfoldl(((k, v),) -> ((f(v), k), (f(v), k)), _rf_findextrema, pairs(domain)) +_rf_findextrema((((fm₁, im₁), (fx₁, ix₁))), (((fm₂, im₂), (fx₂, ix₂)))) = + (isgreater(fm₁, fm₂) ? (fm₂, im₂) : (fm₁, im₁)), (isless(fx₁, fx₂) ? (fx₂, ix₂) : (fx₁, ix₁)) + +""" + findextrema(itr) -> ((mn, index_mn), (mx, index_mx)) + +Return the pair of pairs which would be returned by `(findmin(itr), findmax(itr))`, +but computed in a single pass. + +See also: [`findmin`](@ref), [`findmax`](@ref) + +# Examples + +```jldoctest +julia> findextrema([8, 0.1, -9, pi]) +((-9.0, 3), (8.0, 1)) + +julia> findextrema([1, 7, 7, 6]) +((1, 1), (7, 2)) + +julia> findextrema([1, 7, 7, NaN]) +((NaN, 4), (NaN, 4)) +``` +""" +findextrema(itr) = _findextrema(itr, :) +_findextrema(a, ::Colon) = findextrema(identity, a) + """ argmax(f, domain) diff --git a/base/reducedim.jl b/base/reducedim.jl index dc34b4feb1f6a..76f5c609c6f8b 100644 --- a/base/reducedim.jl +++ b/base/reducedim.jl @@ -1231,6 +1231,153 @@ end reducedim1(R, A) = length(axes1(R)) == 1 + +import Base: promote_op, reduced_indices0, reduced_indices, _findminmax_inittype, check_reducedims, safe_tail, reducedim1, _realtype, axes1, promote_union, isgreater + +function findextrema!(f, op_mn, op_mx, Rval_mn, Rind_mn, Rval_mx, Rind_mx, A::AbstractArray{T,N}) where {T,N} + (isempty(Rval_mn) || isempty(Rval_mx) || isempty(A)) && return ((Rval_mn, Rind_mn), (Rval_mx, Rind_mx)) + lsiz_mn = check_reducedims(Rval_mn, A) + lsiz_mx = check_reducedims(Rval_mx, A) + for i = 1:N + axes(Rval_mn, i) == axes(Rind_mn, i) == axes(Rval_mx, i) == axes(Rind_mx, i) || throw(DimensionMismatch("Find-reduction: outputs must have the same indices")) + end + # Same as findminmax! implementation + indsAt, indsRt = safe_tail(axes(A)), safe_tail(axes(Rval_mn)) + keep, Idefault = Broadcast.shapeindexer(indsRt) + ks = keys(A) + y = iterate(ks) + zi = zero(eltype(ks)) + if reducedim1(Rval_mn, A) + i1 = first(axes1(Rval_mn)) + @inbounds for IA in CartesianIndices(indsAt) + IR = Broadcast.newindex(IA, keep, Idefault) + tmpRv_mn = Rval_mn[i1,IR] + tmpRi_mn = Rind_mn[i1,IR] + tmpRv_mx = Rval_mx[i1,IR] + tmpRi_mx = Rind_mx[i1,IR] + for i in axes(A,1) + k, kss = y::Tuple + tmpAv = f(A[i,IA]) + if tmpRi_mn == zi || op_mn(tmpRv_mn, tmpAv) + tmpRv_mn = tmpAv + tmpRi_mn = k + end + if tmpRi_mx == zi || op_mx(tmpRv_mx, tmpAv) + tmpRv_mx = tmpAv + tmpRi_mx = k + end + y = iterate(ks, kss) + end + Rval_mn[i1,IR] = tmpRv_mn + Rind_mn[i1,IR] = tmpRi_mn + Rval_mx[i1,IR] = tmpRv_mx + Rind_mx[i1,IR] = tmpRi_mx + end + else + @inbounds for IA in CartesianIndices(indsAt) + IR = Broadcast.newindex(IA, keep, Idefault) + for i in axes(A, 1) + k, kss = y::Tuple + tmpAv = f(A[i,IA]) + tmpRv_mn = Rval_mn[i,IR] + tmpRi_mn = Rind_mn[i,IR] + tmpRv_mx = Rval_mx[i,IR] + tmpRi_mx = Rind_mx[i,IR] + if tmpRi_mn == zi || op_mn(tmpRv_mn, tmpAv) + Rval_mn[i,IR] = tmpAv + Rind_mn[i,IR] = k + end + if tmpRi_mx == zi || op_mx(tmpRv_mx, tmpAv) + Rval_mx[i,IR] = tmpAv + Rind_mx[i,IR] = k + end + y = iterate(ks, kss) + end + end + end + ((Rval_mn, Rind_mn), (Rval_mx, Rind_mx)) +end + +""" + findextrema!(rval_mn, rind_mn, rval_mx, rind_mx, A) -> ((minval, index), (maxval, index)) + +Find the minimum and maximum of `A` and the respective linear index along singleton +dimensions, storing the results in `((rval_mn , rind_mn), (rval_mn , rind_mn))`, +equivalent to `(findmin!(rval_mn, rind_mn, A), findmax!(rval_mx, rind_mx, A))` +but computed in a single pass. +""" +function findextrema!(rval_mn::AbstractArray, rind_mn::AbstractArray, rval_mx::AbstractArray, rind_mx::AbstractArray, A::AbstractArray; init::Bool=true) + init && !isempty(A) && (fill!(rval_mn, first(A)); fill!(rval_mx, first(A))) + Ti = eltype(keys(A)) + findextrema!(identity, isgreater, isless, rval_mn, fill!(rind_mn,zero(Ti)), rval_mx, fill!(rind_mx,zero(Ti)), A) +end + +""" + findextrema(A; dims) -> ((minval, index), (maxval, index)) + +For an array input, returns the value and index of the minimum and maximum over the +given dimensions. Equivalent to `(findmin(A; dims), findmax(A; dims))`, but computed +in a single pass. +`NaN` is treated as greater than all other values except `missing`. + +# Examples +```jldoctest +julia> A = [1.0 2; 3 4] +2×2 Matrix{Float64}: + 1.0 2.0 + 3.0 4.0 + +julia> findextrema(A, dims=1) == (findmin(A, dims=1), findmax(A, dims=1)) +true + +julia> findextrema(A, dims=2) == (findmin(A, dims=2), findmax(A, dims=2)) +true +``` +""" +findextrema(A::AbstractArray; dims=:) = _findextrema(A, dims) +_findextrema(A, dims) = _findextrema(identity, A, dims) + +""" + findextrema(f, A; dims) -> ((f(x), index_mn), (f(x), index_mx)) + +For an array input, returns the value in the codomain and index of the corresponding value +which minimize and maximize `f` over the given dimensions. Equivalent to +`(findmin(f, A; dims), findmax(f, A; dims))`, but computed in a single pass. + +# Examples +```jldoctest +julia> A = [-1.0 1; -0.5 2] +2×2 Matrix{Float64}: + -1.0 1.0 + -0.5 2.0 + +julia> findextrema(abs2, A, dims=1) == (findmin(abs2, A, dims=1), findmax(abs2, A, dims=1)) +true + +julia> findextrema(abs2, A, dims=2) == (findmin(abs2, A, dims=2), findmax(abs2, A, dims=2)) +true +``` +""" +findextrema(f, A::AbstractArray; dims=:) = _findextrema(f, A, dims) + +function _findextrema(f, A, region) + ri = reduced_indices0(A, region) + if isempty(A) + if prod(map(length, reduced_indices(A, region))) != 0 + throw(ArgumentError("collection slices must be non-empty")) + end + Tr = promote_op(f, eltype(A)) + Ti = eltype(keys(A)) + (similar(A, Tr, ri), zeros(Ti, ri)), (similar(A, Tr, ri), zeros(Ti, ri)) + else + fA = f(first(A)) + Tr = _findminmax_inittype(f, A) + Ti = eltype(keys(A)) + findextrema!(f, isgreater, isless, fill!(similar(A, Tr, ri), fA), zeros(Ti, ri), + fill!(similar(A, Tr, ri), fA), zeros(Ti, ri), A) + end +end + """ argmin(A; dims) -> indices diff --git a/doc/src/base/collections.md b/doc/src/base/collections.md index 96f540086d021..a8d072678d4db 100644 --- a/doc/src/base/collections.md +++ b/doc/src/base/collections.md @@ -109,8 +109,10 @@ Base.argmax Base.argmin Base.findmax Base.findmin +Base.findextrema Base.findmax! Base.findmin! +Base.findextrema! Base.sum Base.sum! Base.prod diff --git a/test/reduce.jl b/test/reduce.jl index 84d93b12913e4..74709842563bb 100644 --- a/test/reduce.jl +++ b/test/reduce.jl @@ -433,6 +433,17 @@ end @test argmax(sum, Iterators.product(1:5, 1:5)) == (5, 5) end +# findextrema +@testset "findextrema(f, domain)" begin + @test findextrema(-, 1:10) == ((-10, 10), (-1, 1)) + @test findextrema(identity, [1, 2, 3, missing]) === ((missing, 4), (missing, 4)) + @test findextrema(identity, [1, NaN, 3, missing]) === ((missing, 4), (missing, 4)) + @test findextrema(identity, [1, missing, NaN, 3]) === ((missing, 2), (missing, 2)) + @test findextrema(identity, [1, NaN, 3]) === ((NaN, 2), (NaN, 2)) + @test findextrema(identity, [1, 3, NaN]) === ((NaN, 3), (NaN, 3)) + @test findextrema(cos, 0:π/2:2π) == ((-1.0, 3), (1.0, 1)) +end + # any & all @test @inferred any([]) == false diff --git a/test/reducedim.jl b/test/reducedim.jl index 5402376744e82..0c5578b0ba21b 100644 --- a/test/reducedim.jl +++ b/test/reducedim.jl @@ -203,6 +203,21 @@ end @test isequal(f(abs2, A, dims=3), (zeros(Int, 0, 1), zeros(Int, 0, 1))) end + # extrema, findextrema + @test_throws "reducing over an empty collection is not allowed" extrema(A, dims=1) + @test isequal(extrema(A, dims=2), Matrix{Tuple{Int,Int}}(undef, 0,1)) + @test_throws "reducing over an empty collection is not allowed" extrema(A, dims=(1, 2)) + @test isequal(extrema(A, dims=3), Matrix{Tuple{Int,Int}}(undef, 0,1)) + + @test_throws ArgumentError findextrema(A, dims=1) + @test isequal(findextrema(A, dims=2), ((zeros(Int, 0, 1), zeros(Int, 0, 1)), (zeros(Int, 0, 1), zeros(Int, 0, 1)))) + @test_throws ArgumentError findextrema(A, dims=(1, 2)) + @test isequal(findextrema(A, dims=3), ((zeros(Int, 0, 1), zeros(Int, 0, 1)), (zeros(Int, 0, 1), zeros(Int, 0, 1)))) + @test_throws ArgumentError findextrema(abs2, A, dims=1) + @test isequal(findextrema(abs2, A, dims=2), ((zeros(Int, 0, 1), zeros(Int, 0, 1)), (zeros(Int, 0, 1), zeros(Int, 0, 1)))) + @test_throws ArgumentError findextrema(abs2, A, dims=(1, 2)) + @test isequal(findextrema(abs2, A, dims=3), ((zeros(Int, 0, 1), zeros(Int, 0, 1)), (zeros(Int, 0, 1), zeros(Int, 0, 1)))) + end ## findmin/findmax/minimum/maximum @@ -229,6 +244,10 @@ for (tup, rval, rind) in [((1,), [5.0 5.0 6.0], [CartesianIndex(2,1) CartesianIn @test isequal(maximum!(copy(rval), A, init=false), rval) end +for tup in [(1,), (2,), (1,2)] + @test findextrema(A, dims=tup) == (findmin(A, dims=tup), findmax(A, dims=tup)) +end + @testset "findmin/findmax transformed arguments, numeric values" begin A = [1.0 -5.0 -6.0; -5.0 2.0 4.0] @@ -249,6 +268,8 @@ end @test rind′ == rind @test findmin(f, A, dims=tup) == (rval, rind) @test (rval′, rind′) == findmin(A′, dims=tup) + @test findextrema(A′, dims=tup)[1] == (rval, rind) + @test findextrema(f, A, dims=tup)[1] == (rval, rind) end end @@ -269,6 +290,8 @@ end @test rind′ == rind @test findmax(f, A, dims=tup) == (rval, rind) @test (rval′, rind′) == findmax(A′, dims=tup) + @test findextrema(A′, dims=tup)[2] == (rval, rind) + @test findextrema(f, A, dims=tup)[2] == (rval, rind) end end end @@ -282,6 +305,9 @@ end rval′, rind′ = findmin(length, A, dims=tup) @test (rval, rind) == (rval′, rind′) @test typeof(rval′) == Matrix{Int} + rval′, rind′ = findextrema(length, A, dims=tup)[1] + @test (rval, rind) == (rval′, rind′) + @test typeof(rval′) == Matrix{Int} end for (tup, rval, rind) in [((1,), [3 4], [CartesianIndex(2, 1) CartesianIndex(2, 2)]), ((2,), reshape([2, 4], 2, 1), reshape([CartesianIndex(1, 2), CartesianIndex(2, 2)], 2, 1)), @@ -289,6 +315,9 @@ end rval′, rind′ = findmax(length, A, dims=tup) @test (rval, rind) == (rval′, rind′) @test typeof(rval) == Matrix{Int} + rval′, rind′ = findextrema(length, A, dims=tup)[2] + @test (rval, rind) == (rval′, rind′) + @test typeof(rval′) == Matrix{Int} end B = [1.5 1.0; 5.5 6.0] for (tup, rval, rind) in [((1,), [3//2 1//1], [CartesianIndex(1, 1) CartesianIndex(1, 2)]), @@ -297,9 +326,15 @@ end rval′, rind′ = findmin(Rational, B, dims=tup) @test (rval, rind) == (rval′, rind′) @test typeof(rval) == Matrix{Rational{Int}} + rval′, rind′ = findextrema(Rational, B, dims=tup)[1] + @test (rval, rind) == (rval′, rind′) + @test typeof(rval) == Matrix{Rational{Int}} rval′, rind′ = findmin(Rational ∘ abs ∘ complex, B, dims=tup) @test (rval, rind) == (rval′, rind′) @test typeof(rval) == Matrix{Rational{Int}} + rval′, rind′ = findextrema(Rational ∘ abs ∘ complex, B, dims=tup)[1] + @test (rval, rind) == (rval′, rind′) + @test typeof(rval) == Matrix{Rational{Int}} end end @@ -316,6 +351,7 @@ end @test all(rind′ .== rind) @test all(maximum(B, dims=tup) .=== rval) @test isequal(findmax(abs, B′, dims=tup), (rval′, rind′)) + @test isequal(findextrema(abs, B′, dims=tup)[2], (rval′, rind′)) end for (tup, rval, rind) in [(1, [1.0 missing missing], [CartesianIndex(1, 1) CartesianIndex(1, 2) CartesianIndex(2, 3)]), @@ -325,6 +361,7 @@ end @test all(rind′ .== rind) @test all(minimum(B, dims=tup) .=== rval) @test isequal(findmin(abs, B′, dims=tup), (rval′, rind′)) + @test isequal(findextrema(abs, B′, dims=tup)[1], (rval′, rind′)) end end @@ -351,6 +388,8 @@ for (tup, rval, rind) in [((1,), [NaN 2.0 4.0], [CartesianIndex(2,1) CartesianIn @test isequal(findmin(A, dims=tup), (rval, rind)) @test isequal(findmin(abs, A, dims=tup), (rval, rind)) @test isequal(findmin!(similar(rval), similar(rind), A), (rval, rind)) + @test isequal(findextrema(A, dims=tup)[1], (rval, rind)) + @test isequal(findextrema(abs, A, dims=tup)[1], (rval, rind)) @test isequal(minimum(A, dims=tup), rval) @test isequal(minimum!(similar(rval), A), rval) @test isequal(minimum!(copy(rval), A, init=false), rval) @@ -363,6 +402,8 @@ for (tup, rval, rind) in [((1,), [NaN 3.0 6.0], [CartesianIndex(2,1) CartesianIn @test isequal(findmax(A, dims=tup), (rval, rind)) @test isequal(findmax(abs, A, dims=tup), (rval, rind)) @test isequal(findmax!(similar(rval), similar(rind), A), (rval, rind)) + @test isequal(findextrema(A, dims=tup)[2], (rval, rind)) + @test isequal(findextrema(abs, A, dims=tup)[2], (rval, rind)) @test isequal(maximum(A, dims=tup), rval) @test isequal(maximum!(similar(rval), A), rval) @test isequal(maximum!(copy(rval), A, init=false), rval) @@ -379,6 +420,10 @@ end @test minimum(abs2, A, dims=2) == reshape([0.25], 1, 1) @test findmin(abs2, A, dims=2) == (fill(0.25, 1, 1), fill(CartesianIndex(1, 2), 1, 1)) @test findmax(abs2, A, dims=2) == (fill(2.25, 1, 1), fill(CartesianIndex(1, 1), 1, 1)) + mn, mx = findextrema(abs2, A, dims=2) + @test mn == (fill(0.25, 1, 1), fill(CartesianIndex(1, 2), 1, 1)) + @test mx == (fill(2.25, 1, 1), fill(CartesianIndex(1, 1), 1, 1)) + # @test findextrema(abs2, A, dims=2) == ((fill(0.25, 1, 1), fill(CartesianIndex(1, 2), 1, 1)), (fill(2.25, 1, 1), fill(CartesianIndex(1, 1), 1, 1))) end end @@ -393,6 +438,9 @@ end @test isequal(findmin(A, dims=tup), (rval, rind)) @test isequal(findmin(abs, A′, dims=tup), (rval, rind)) @test isequal(findmin!(similar(rval), similar(rind), A), (rval, rind)) + @test isequal(findextrema(A, dims=tup)[1], (rval, rind)) + @test isequal(findextrema(abs, A, dims=tup)[1], (rval, rind)) + @test isequal(findextrema!(similar(rval), similar(rind), similar(rval), similar(rind), A)[1], (rval, rind)) @test isequal(minimum(A, dims=tup), rval) @test isequal(minimum!(similar(rval), A), rval) @test isequal(minimum!(copy(rval), A, init=false), rval) @@ -404,6 +452,9 @@ end @test isequal(findmax(A, dims=tup), (rval, rind)) @test isequal(findmax(abs, A′, dims=tup), (rval, rind)) @test isequal(findmax!(similar(rval), similar(rind), A), (rval, rind)) + @test isequal(findextrema(A, dims=tup)[2], (rval, rind)) + @test isequal(findextrema(abs, A, dims=tup)[2], (rval, rind)) + @test isequal(findextrema!(similar(rval), similar(rind), similar(rval), similar(rind), A)[2], (rval, rind)) @test isequal(maximum(A, dims=tup), rval) @test isequal(maximum!(similar(rval), A), rval) @test isequal(maximum!(copy(rval), A, init=false), rval) @@ -421,6 +472,9 @@ end @test isequal(findmin(A, dims=tup), (rval, rind)) @test isequal(findmin(x -> x == 1 ? Inf : -Inf, A′, dims=tup), (rval, rind)) @test isequal(findmin!(similar(rval), similar(rind), A), (rval, rind)) + @test isequal(findextrema(A, dims=tup)[1], (rval, rind)) + @test isequal(findextrema(x -> x == 1 ? Inf : -Inf, A′, dims=tup)[1], (rval, rind)) + @test isequal(findextrema!(similar(rval), similar(rind), similar(rval), similar(rind), A)[1], (rval, rind)) @test isequal(minimum(A, dims=tup), rval) @test isequal(minimum!(similar(rval), A), rval) @test isequal(minimum!(copy(rval), A, init=false), rval) @@ -432,6 +486,9 @@ end @test isequal(findmax(A, dims=tup), (rval, rind)) @test isequal(findmax(x -> x == 1 ? Inf : -Inf, A′, dims=tup), (rval, rind)) @test isequal(findmax!(similar(rval), similar(rind), A), (rval, rind)) + @test isequal(findextrema(A, dims=tup)[2], (rval, rind)) + @test isequal(findextrema(x -> x == 1 ? Inf : -Inf, A′, dims=tup)[2], (rval, rind)) + @test isequal(findextrema!(similar(rval), similar(rind), similar(rval), similar(rind), A)[2], (rval, rind)) @test isequal(maximum(A, dims=tup), rval) @test isequal(maximum!(similar(rval), A), rval) @test isequal(maximum!(copy(rval), A, init=false), rval) @@ -445,6 +502,9 @@ end @test isequal(findmin(A, dims=tup), (rval, rind)) @test isequal(findmin(x -> 10^x, A′, dims=tup), (rval, rind)) @test isequal(findmin!(similar(rval), similar(rind), A), (rval, rind)) + @test isequal(findextrema(A, dims=tup)[1], (rval, rind)) + @test isequal(findextrema(x -> 10^x, A′, dims=tup)[1], (rval, rind)) + @test isequal(findextrema!(similar(rval), similar(rind), similar(rval), similar(rind), A)[1], (rval, rind)) @test isequal(minimum(A, dims=tup), rval) @test isequal(minimum!(similar(rval), A), rval) @test isequal(minimum!(copy(rval), A, init=false), rval) @@ -454,6 +514,9 @@ end @test isequal(findmax(A, dims=tup), (rval, rind)) @test isequal(findmax(x -> 10^x, A′, dims=tup), (rval, rind)) @test isequal(findmax!(similar(rval), similar(rind), A), (rval, rind)) + @test isequal(findextrema(A, dims=tup)[2], (rval, rind)) + @test isequal(findextrema(x -> 10^x, A′, dims=tup)[2], (rval, rind)) + @test isequal(findextrema!(similar(rval), similar(rind), similar(rval), similar(rind), A)[2], (rval, rind)) @test isequal(maximum(A, dims=tup), rval) @test isequal(maximum!(similar(rval), A), rval) @test isequal(maximum!(copy(rval), A, init=false), rval) @@ -464,6 +527,9 @@ end @test isequal(findmin(A, dims=tup), (rval, rind)) @test isequal(findmin(x -> -(x + 20), A, dims=tup), (rval, rind)) @test isequal(findmin!(similar(rval), similar(rind), A), (rval, rind)) + @test isequal(findextrema(A, dims=tup)[1], (rval, rind)) + @test isequal(findextrema(x -> -(x + 20), A, dims=tup)[1], (rval, rind)) + @test isequal(findextrema!(similar(rval), similar(rind), similar(rval), similar(rind), A)[1], (rval, rind)) @test isequal(minimum(A, dims=tup), rval) @test isequal(minimum!(similar(rval), A), rval) @test isequal(minimum!(copy(rval), A, init=false), rval) @@ -473,6 +539,9 @@ end @test isequal(findmax(A, dims=tup), (rval, rind)) @test isequal(findmax(x -> -(x + 20), A, dims=tup), (rval, rind)) @test isequal(findmax!(similar(rval), similar(rind), A), (rval, rind)) + @test isequal(findextrema(A, dims=tup)[2], (rval, rind)) + @test isequal(findextrema(x -> -(x + 20), A, dims=tup)[2], (rval, rind)) + @test isequal(findextrema!(similar(rval), similar(rind), similar(rval), similar(rind), A)[2], (rval, rind)) @test isequal(maximum(A, dims=tup), rval) @test isequal(maximum!(similar(rval), A), rval) @test isequal(maximum!(copy(rval), A, init=false), rval) @@ -484,6 +553,9 @@ end @test isequal(findmin(A, dims=tup), (rval, rind)) @test isequal(findmin(x -> x == 1 ? 10^x : x - 20, A′, dims=tup), (rval, rind)) @test isequal(findmin!(similar(rval), similar(rind), A), (rval, rind)) + @test isequal(findextrema(A, dims=tup)[1], (rval, rind)) + @test isequal(findextrema(x -> x == 1 ? 10^x : x - 20, A′, dims=tup)[1], (rval, rind)) + @test isequal(findextrema!(similar(rval), similar(rind), similar(rval), similar(rind), A)[1], (rval, rind)) @test isequal(minimum(A, dims=tup), rval) @test isequal(minimum!(similar(rval), A), rval) @test isequal(minimum!(copy(rval), A, init=false), rval) @@ -493,6 +565,9 @@ end @test isequal(findmax(A, dims=tup), (rval, rind)) @test isequal(findmax(x -> x == 1 ? 10^x : x - 20, A′, dims=tup), (rval, rind)) @test isequal(findmax!(similar(rval), similar(rind), A), (rval, rind)) + @test isequal(findextrema(A, dims=tup)[2], (rval, rind)) + @test isequal(findextrema(x -> x == 1 ? 10^x : x - 20, A′, dims=tup)[2], (rval, rind)) + @test isequal(findextrema!(similar(rval), similar(rind), similar(rval), similar(rind), A)[2], (rval, rind)) @test isequal(maximum(A, dims=tup), rval) @test isequal(maximum!(similar(rval), A), rval) @test isequal(maximum!(copy(rval), A, init=false), rval) @@ -505,6 +580,9 @@ end @test isequal(findmin(A, dims=tup), (rval, rind)) @test isequal(findmin(x -> (x^2)[1:1], A, dims=tup), (rval, rind)) @test isequal(findmin!(similar(rval), similar(rind), A), (rval, rind)) + @test isequal(findextrema(A, dims=tup)[1], (rval, rind)) + @test isequal(findextrema(x -> (x^2)[1:1], A, dims=tup)[1], (rval, rind)) + @test isequal(findextrema!(similar(rval), similar(rind), similar(rval), similar(rind), A)[1], (rval, rind)) @test isequal(minimum(A, dims=tup), rval) @test isequal(minimum!(similar(rval), A), rval) @test isequal(minimum!(copy(rval), A, init=false), rval) @@ -514,6 +592,9 @@ end @test isequal(findmax(A, dims=tup), (rval, rind)) @test isequal(findmax(x -> (x^2)[1:1], A, dims=tup), (rval, rind)) @test isequal(findmax!(similar(rval), similar(rind), A), (rval, rind)) + @test isequal(findextrema(A, dims=tup)[2], (rval, rind)) + @test isequal(findextrema(x -> (x^2)[1:1], A, dims=tup)[2], (rval, rind)) + @test isequal(findextrema!(similar(rval), similar(rind), similar(rval), similar(rind), A)[2], (rval, rind)) @test isequal(maximum(A, dims=tup), rval) @test isequal(maximum!(similar(rval), A), rval) @test isequal(maximum!(copy(rval), A, init=false), rval)