diff --git a/NEWS.md b/NEWS.md index 68bf8c098e3020..17a6e8346f00ea 100644 --- a/NEWS.md +++ b/NEWS.md @@ -468,6 +468,12 @@ Deprecated or removed * The timing functions `tic`, `toc`, and `toq` are deprecated in favor of `@time` and `@elapsed` ([#17046]). + * Methods of `findfirst`, `findnext`, `findlast`, and `findprev` that accept a value to + search for are deprecated in favor of passing a predicate ([#19186], [#10593]). + + * `find` functions now operate only on booleans by default. To look for non-zeros, use + `x->x!=0` or `!iszero` ([#23120]). + Command-line option changes --------------------------- diff --git a/base/array.jl b/base/array.jl index 64189afa332193..e1ada9d7e661e9 100644 --- a/base/array.jl +++ b/base/array.jl @@ -1596,14 +1596,14 @@ cat(n::Integer, x::Integer...) = reshape([x...], (ntuple(x->1, n-1)..., length(x """ findnext(A, i::Integer) -Find the next linear index >= `i` of a non-zero element of `A`, or `0` if not found. +Find the next linear index >= `i` of a true element of `A`, or `0` if not found. # Examples ```jldoctest -julia> A = [0 0; 1 0] -2×2 Array{Int64,2}: - 0 0 - 1 0 +julia> A = [false false; true false] +2×2 Array{Bool,2}: + false false + true false julia> findnext(A,1) 2 @@ -1615,8 +1615,14 @@ julia> findnext(A,3) function findnext(A, start::Integer) l = endof(A) i = start + warned = false while i <= l - if A[i] != 0 + a = A[i] + if !warned && !(a isa Bool) + depwarn("In the future `findnext` will only work on boolean collections. Use `findnext(x->x!=0, A)` instead.", :findnext) + warned = true + end + if a != 0 return i end i = nextind(A, i) @@ -1646,58 +1652,6 @@ julia> findfirst(zeros(3)) """ findfirst(A) = findnext(A, 1) -""" - findnext(A, v, i::Integer) - -Find the next linear index >= `i` of an element of `A` equal to `v` (using `==`), or `0` if not found. - -# Examples -```jldoctest -julia> A = [1 4; 2 2] -2×2 Array{Int64,2}: - 1 4 - 2 2 - -julia> findnext(A,4,4) -0 - -julia> findnext(A,4,3) -3 -``` -""" -function findnext(A, v, start::Integer) - l = endof(A) - i = start - while i <= l - if A[i] == v - return i - end - i = nextind(A, i) - end - return 0 -end -""" - findfirst(A, v) - -Return the linear index of the first element equal to `v` in `A`. -Returns `0` if `v` is not found. - -# Examples -```jldoctest -julia> A = [4 6; 2 2] -2×2 Array{Int64,2}: - 4 6 - 2 2 - -julia> findfirst(A,2) -2 - -julia> findfirst(A,3) -0 -``` -""" -findfirst(A, v) = findnext(A, v, 1) - """ findnext(predicate::Function, A, i::Integer) @@ -1754,14 +1708,14 @@ findfirst(testf::Function, A) = findnext(testf, A, 1) """ findprev(A, i::Integer) -Find the previous linear index <= `i` of a non-zero element of `A`, or `0` if not found. +Find the previous linear index <= `i` of a true element of `A`, or `0` if not found. # Examples ```jldoctest -julia> A = [0 0; 1 2] -2×2 Array{Int64,2}: - 0 0 - 1 2 +julia> A = [false false; true true] +2×2 Array{Bool,2}: + false false + true true julia> findprev(A,2) 2 @@ -1772,8 +1726,14 @@ julia> findprev(A,1) """ function findprev(A, start::Integer) i = start + warned = false while i >= 1 - A[i] != 0 && return i + a = A[i] + if !warned && !(a isa Bool) + depwarn("In the future `findprev` will only work on boolean collections. Use `findprev(x->x!=0, A)` instead.", :findprev) + warned = true + end + a != 0 && return i i = prevind(A, i) end return 0 @@ -1806,59 +1766,6 @@ julia> findlast(A) """ findlast(A) = findprev(A, endof(A)) -""" - findprev(A, v, i::Integer) - -Find the previous linear index <= `i` of an element of `A` equal to `v` (using `==`), or `0` if not found. - -# Examples -```jldoctest -julia> A = [0 0; 1 2] -2×2 Array{Int64,2}: - 0 0 - 1 2 - -julia> findprev(A, 1, 4) -2 - -julia> findprev(A, 1, 1) -0 -``` -""" -function findprev(A, v, start::Integer) - i = start - while i >= 1 - A[i] == v && return i - i = prevind(A, i) - end - return 0 -end - -""" - findlast(A, v) - -Return the linear index of the last element equal to `v` in `A`. -Returns `0` if there is no element of `A` equal to `v`. - -# Examples -```jldoctest -julia> A = [1 2; 2 1] -2×2 Array{Int64,2}: - 1 2 - 2 1 - -julia> findlast(A,1) -4 - -julia> findlast(A,2) -3 - -julia> findlast(A,3) -0 -``` -""" -findlast(A, v) = findprev(A, v, endof(A)) - """ findprev(predicate::Function, A, i::Integer) @@ -1952,9 +1859,7 @@ _index_remapper(iter) = OneTo(typemax(Int)) # safe for objects that don't imple """ find(A) -Return a vector of the linear indexes of the non-zeros in `A` (determined by `A[i]!=0`). A -common use of this is to convert a boolean array to an array of indexes of the `true` -elements. If there are no non-zero elements of `A`, `find` returns an empty array. +Return a vector of the linear indices of the true values in `A`. # Examples ```jldoctest @@ -1968,7 +1873,7 @@ julia> find(A) 1 4 -julia> find(zeros(3)) +julia> find(falses(3)) 0-element Array{Int64,1} ``` """ @@ -1977,7 +1882,12 @@ function find(A) I = Vector{Int}(nnzA) cnt = 1 inds = _index_remapper(A) + warned = false for (i,a) in enumerate(A) + if !warned && !(a isa Bool) + depwarn("In the future `find(A)` will only work on boolean collections. Use `find(x->x!=0, A)` instead.", :find) + warned = true + end if a != 0 I[cnt] = inds[i] cnt += 1 @@ -1986,7 +1896,7 @@ function find(A) return I end -find(x::Number) = x == 0 ? Array{Int,1}(0) : [1] +find(x::Bool) = x ? [1] : Array{Int,1}(0) find(testf::Function, x::Number) = !testf(x) ? Array{Int,1}(0) : [1] findn(A::AbstractVector) = find(A) diff --git a/base/combinatorics.jl b/base/combinatorics.jl index 28e0e451b89b7c..21dbbb4a576093 100644 --- a/base/combinatorics.jl +++ b/base/combinatorics.jl @@ -75,7 +75,7 @@ function permute!!(a, p::AbstractVector{<:Integer}) count = 0 start = 0 while count < length(a) - ptr = start = findnext(p, start+1) + ptr = start = findnext(equalto(start+1), p) temp = a[start] next = p[start] count += 1 @@ -125,7 +125,7 @@ function ipermute!!(a, p::AbstractVector{<:Integer}) count = 0 start = 0 while count < length(a) - start = findnext(p, start+1) + start = findnext(equalto(start+1), p) temp = a[start] next = p[start] count += 1 diff --git a/base/datafmt.jl b/base/datafmt.jl index 6ceecd14acf404..173565e8154e8a 100644 --- a/base/datafmt.jl +++ b/base/datafmt.jl @@ -370,7 +370,7 @@ function val_opts(opts) for (opt_name, opt_val) in opts in(opt_name, valid_opts) || throw(ArgumentError("unknown option $opt_name")) - opt_typ = valid_opt_types[findfirst(valid_opts, opt_name)] + opt_typ = valid_opt_types[findfirst(Base.equalto(opt_name), valid_opts)] isa(opt_val, opt_typ) || throw(ArgumentError("$opt_name should be of type $opt_typ, got $(typeof(opt_val))")) d[opt_name] = opt_val diff --git a/base/deprecated.jl b/base/deprecated.jl index 34176d7259858e..7d6c8d32a87ad8 100644 --- a/base/deprecated.jl +++ b/base/deprecated.jl @@ -1855,6 +1855,12 @@ end nothing end +@deprecate find(x::Number) find(x->x!=0, x) +@deprecate findnext(A, v, i::Integer) findnext(x->x==v, A, i) +@deprecate findfirst(A, v) findfirst(x->x==v, A) +@deprecate findprev(A, v, i::Integer) findprev(x->x==v, A, i) +@deprecate findlast(A, v) findlast(x->x==v, A) + # END 0.7 deprecations # BEGIN 1.0 deprecations diff --git a/base/event.jl b/base/event.jl index 74bb6ba3069a32..57ccd302a06927 100644 --- a/base/event.jl +++ b/base/event.jl @@ -234,7 +234,7 @@ function ensure_rescheduled(othertask::Task) # if the current task was queued, # also need to return it to the runnable state # before throwing an error - i = findfirst(Workqueue, ct) + i = findfirst(t->t===ct, Workqueue) i == 0 || deleteat!(Workqueue, i) ct.state = :runnable end diff --git a/base/file.jl b/base/file.jl index 9876aeb5ac89a1..af6d8467b89b69 100644 --- a/base/file.jl +++ b/base/file.jl @@ -271,7 +271,7 @@ function tempname(temppath::AbstractString,uunique::UInt32) tempp = cwstring(temppath) tname = Vector{UInt16}(32767) uunique = ccall(:GetTempFileNameW,stdcall,UInt32,(Ptr{UInt16},Ptr{UInt16},UInt32,Ptr{UInt16}), tempp,temp_prefix,uunique,tname) - lentname = findfirst(tname,0)-1 + lentname = findfirst(Base.equalto(0),tname)-1 if uunique == 0 || lentname <= 0 error("GetTempFileName failed: $(Libc.FormatMessage())") end diff --git a/base/inference.jl b/base/inference.jl index ec9ac4b9536d09..a908ba9e191614 100644 --- a/base/inference.jl +++ b/base/inference.jl @@ -1583,7 +1583,7 @@ function builtin_tfunction(@nospecialize(f), argtypes::Array{Any,1}, end tf = t_ifunc[iidx] else - fidx = findfirst(t_ffunc_key, f) + fidx = findfirst(x->x===f, t_ffunc_key) if fidx == 0 # unknown/unhandled builtin function return Any @@ -4757,7 +4757,7 @@ function statement_cost(ex::Expr, line::Int, src::CodeInfo, mod::Module, params: elseif f == Main.Core.arrayref return plus_saturate(argcost, isknowntype(ex.typ) ? 4 : params.inline_nonleaf_penalty) end - fidx = findfirst(t_ffunc_key, f) + fidx = findfirst(x->x===f, t_ffunc_key) if fidx == 0 # unknown/unhandled builtin or anonymous function # Use the generic cost of a direct function call @@ -5604,7 +5604,7 @@ function _getfield_elim_pass!(e::Expr, sv::InferenceState) if alloc !== false flen, fnames = alloc if isa(j, QuoteNode) - j = findfirst(fnames, j.value) + j = findfirst(x->x == j.value, fnames) end if 1 <= j <= flen ok = true @@ -5899,7 +5899,7 @@ function replace_getfield!(e::Expr, tupname, vals, field_names, sv::InferenceSta a.args[3] else @assert isa(a.args[3], QuoteNode) - findfirst(field_names, a.args[3].value) + findfirst(x->x == a.args[3].value, field_names) end @assert(idx > 0) # clients should check that all getfields are valid val = vals[idx] diff --git a/base/operators.jl b/base/operators.jl index 138e6b7c2dcf10..3a16be76e17198 100644 --- a/base/operators.jl +++ b/base/operators.jl @@ -972,3 +972,5 @@ julia> filter(!isalpha, str) ``` """ !(f::Function) = (x...)->!f(x...) + +equalto(y) = x->isequal(x,y) diff --git a/base/permuteddimsarray.jl b/base/permuteddimsarray.jl index cc91063e5108b8..cb0ccee96cc13c 100644 --- a/base/permuteddimsarray.jl +++ b/base/permuteddimsarray.jl @@ -148,7 +148,7 @@ function _copy!(P::PermutedDimsArray{T,N,perm}, src) where {T,N,perm} copy!(parent(P), src) # it's not permuted else R1 = CartesianRange(indices(src)[1:d]) - d1 = findfirst(perm, d+1) # first permuted dim of dest + d1 = findfirst(Base.equalto(d+1), perm) # first permuted dim of dest R2 = CartesianRange(indices(src)[d+2:d1-1]) R3 = CartesianRange(indices(src)[d1+1:end]) _permutedims!(P, src, R1, R2, R3, d+1, d1) diff --git a/base/pkg/query.jl b/base/pkg/query.jl index dbe1919f0664d8..c0f588a73149a7 100644 --- a/base/pkg/query.jl +++ b/base/pkg/query.jl @@ -359,7 +359,7 @@ function prune_versions(reqs::Requires, deps::Dict{String,Dict{VersionNumber,Ava vmaskp[vn] = falses(luds) end for (vn,a) in fdepsp - vmind = findfirst(uniqdepssets, a.requires) + vmind = findfirst(Base.equalto(a.requires), uniqdepssets) @assert vmind > 0 vm = vmaskp[vn] vm[vmind] = true @@ -389,7 +389,7 @@ function prune_versions(reqs::Requires, deps::Dict{String,Dict{VersionNumber,Ava nc = length(vmask0_uniq) classes = [VersionNumber[] for c0 = 1:nc] for (vn,vm) in vmaskp - c0 = findfirst(vmask0_uniq, vm) + c0 = findfirst(Base.equalto(vm), vmask0_uniq) push!(classes[c0], vn) end map(sort!, classes) diff --git a/base/precompile.jl b/base/precompile.jl index 4981c89231dd3a..86bcc216f32c83 100644 --- a/base/precompile.jl +++ b/base/precompile.jl @@ -819,7 +819,6 @@ precompile(Tuple{Type{BoundsError}, Array{Int64, 2}, Tuple{Base.UnitRange{Int64} precompile(Tuple{typeof(Base.throw_boundserror), Array{Int64, 2}, Tuple{Base.UnitRange{Int64}, Int64}}) precompile(Tuple{getfield(Base.Cartesian, Symbol("#@nexprs")), Int64, Expr}) precompile(Tuple{typeof(Base.Cartesian._nexprs), Int64, Expr}) -precompile(Tuple{typeof(Core.Inference.findnext), Array{Function, 1}, typeof(===), Int64}) precompile(Tuple{typeof(Core.Inference.builtin_tfunction), typeof(===), Array{Any, 1}, Core.Inference.InferenceState, Core.Inference.InferenceParams}) precompile(Tuple{typeof(Core.Inference.typeinf_frame), Core.MethodInstance, Bool, Bool, Core.Inference.InferenceParams}) precompile(Tuple{typeof(Core.Inference.typeinf), Core.Inference.InferenceState}) diff --git a/base/repl/LineEdit.jl b/base/repl/LineEdit.jl index 7c330774fef991..a8b898c3d0f06d 100644 --- a/base/repl/LineEdit.jl +++ b/base/repl/LineEdit.jl @@ -620,10 +620,10 @@ const _space = UInt8(' ') _notspace(c) = c != _space -beginofline(buf, pos=position(buf)) = findprev(buf.data, _newline, pos) +beginofline(buf, pos=position(buf)) = findprev(Base.equalto(_newline), buf.data, pos) function endofline(buf, pos=position(buf)) - eol = findnext(buf.data[pos+1:buf.size], _newline, 1) + eol = findnext(Base.equalto(_newline), buf.data[pos+1:buf.size], 1) eol == 0 ? buf.size : pos + eol - 1 end diff --git a/base/sharedarray.jl b/base/sharedarray.jl index 3cdeb3bcd88a1e..4c3d24a5b31efc 100644 --- a/base/sharedarray.jl +++ b/base/sharedarray.jl @@ -390,7 +390,7 @@ sub_1dim(S::SharedArray, pidx) = view(S.s, range_1dim(S, pidx)) function init_loc_flds(S::SharedArray{T,N}, empty_local=false) where T where N if myid() in S.pids - S.pidx = findfirst(S.pids, myid()) + S.pidx = findfirst(Base.equalto(myid()), S.pids) if isa(S.refs[1], Future) refid = remoteref_id(S.refs[S.pidx]) else diff --git a/base/sparse/sparsevector.jl b/base/sparse/sparsevector.jl index 95c50798f3cdbd..e6279f2998d619 100644 --- a/base/sparse/sparsevector.jl +++ b/base/sparse/sparsevector.jl @@ -1890,7 +1890,7 @@ function sort(x::SparseVector{Tv,Ti}; kws...) where {Tv,Ti} allvals = push!(copy(nonzeros(x)),zero(Tv)) sinds = sortperm(allvals;kws...) n,k = length(x),length(allvals) - z = findfirst(sinds,k) + z = findfirst(Base.equalto(k),sinds) newnzind = collect(Ti,1:k-1) newnzind[z:end] .+= n-k+1 newnzvals = allvals[deleteat!(sinds[1:k],z)] diff --git a/doc/src/stdlib/arrays.md b/doc/src/stdlib/arrays.md index 1f6a8d0c4f7e45..27dc55340d9d20 100644 --- a/doc/src/stdlib/arrays.md +++ b/doc/src/stdlib/arrays.md @@ -111,17 +111,13 @@ Base.find(::Function, ::Any) Base.findn Base.findnz Base.findfirst(::Any) -Base.findfirst(::Any, ::Any) Base.findfirst(::Function, ::Any) Base.findlast(::Any) -Base.findlast(::Any, ::Any) Base.findlast(::Function, ::Any) Base.findnext(::Any, ::Integer) Base.findnext(::Function, ::Any, ::Integer) -Base.findnext(::Any, ::Any, ::Integer) Base.findprev(::Any, ::Integer) Base.findprev(::Function, ::Any, ::Integer) -Base.findprev(::Any, ::Any, ::Integer) Base.permutedims Base.permutedims! Base.PermutedDimsArray diff --git a/test/arrayops.jl b/test/arrayops.jl index 5d91b9f32bfea2..7165de1b5598f3 100644 --- a/test/arrayops.jl +++ b/test/arrayops.jl @@ -427,44 +427,43 @@ end @test X[Y[end],1] == 5 @test X[end,Y[end]] == 11 end +equalto(y) = x->x==y @testset "find, findfirst, findnext, findlast, findprev" begin a = [0,1,2,3,0,1,2,3] - @test find(a) == [2,3,4,6,7,8] + @test find(!equalto(0), a) == [2,3,4,6,7,8] @test find(a.==2) == [3,7] @test find(isodd,a) == [2,4,6,8] - @test findfirst(a) == 2 + @test findfirst(!equalto(0), a) == 2 @test findfirst(a.==0) == 1 @test findfirst(a.==5) == 0 - @test findfirst([1,2,4,1,2,3,4], 3) == 6 + @test findfirst(equalto(3), [1,2,4,1,2,3,4]) == 6 @test findfirst(isodd, [2,4,6,3,9,2,0]) == 4 @test findfirst(isodd, [2,4,6,2,0]) == 0 - @test findnext(a,4) == 4 - @test findnext(a,5) == 6 - @test findnext(a,1) == 2 - @test findnext(a,1,4) == 6 - @test findnext(a,5,4) == 0 - @test findlast(a) == 8 + @test findnext(!equalto(0),a,4) == 4 + @test findnext(!equalto(0),a,5) == 6 + @test findnext(!equalto(0),a,1) == 2 + @test findnext(equalto(1),a,4) == 6 + @test findnext(equalto(5),a,4) == 0 + @test findlast(!equalto(0), a) == 8 @test findlast(a.==0) == 5 @test findlast(a.==5) == 0 - @test findlast([1,2,4,1,2,3,4], 3) == 6 + @test findlast(equalto(3), [1,2,4,1,2,3,4]) == 6 @test findlast(isodd, [2,4,6,3,9,2,0]) == 5 @test findlast(isodd, [2,4,6,2,0]) == 0 - @test findprev(a,4) == 4 - @test findprev(a,5) == 4 - @test findprev(a,1) == 0 - @test findprev(a,1,4) == 2 - @test findprev(a,1,8) == 6 + @test findprev(!equalto(0),a,4) == 4 + @test findprev(!equalto(0),a,5) == 4 + @test findprev(!equalto(0),a,1) == 0 + @test findprev(equalto(1),a,4) == 2 + @test findprev(equalto(1),a,8) == 6 @test findprev(isodd, [2,4,5,3,9,2,0], 7) == 5 @test findprev(isodd, [2,4,5,3,9,2,0], 2) == 0 end @testset "find with general iterables" begin s = "julia" - # FIXME once 16269 is resolved - # @test find(s) == [1,2,3,4,5] @test find(c -> c == 'l', s) == [3] g = graphemes("日本語") @test find(isascii, g) == Int[] - @test find((i % 2 for i in 1:10)) == collect(1:2:9) + @test find(!equalto(0), (i % 2 for i in 1:10)) == collect(1:2:9) end @testset "findn" begin b = findn(ones(2,2,2,2)) diff --git a/test/inference.jl b/test/inference.jl index 6d8e5cf78ac203..81910152cf37ec 100644 --- a/test/inference.jl +++ b/test/inference.jl @@ -1102,7 +1102,7 @@ isdefined_f3(x) = isdefined(x, 3) @test find_call(first(code_typed(isdefined_f3, Tuple{Tuple{Vararg{Int}}})[1]).code, isdefined, 3) let isa_tfunc = Core.Inference.t_ffunc_val[ - findfirst(Core.Inference.t_ffunc_key, isa)][3] + findfirst(x->x===isa, Core.Inference.t_ffunc_key)][3] @test isa_tfunc(Array, Const(AbstractArray)) === Const(true) @test isa_tfunc(Array, Type{AbstractArray}) === Const(true) @test isa_tfunc(Array, Type{AbstractArray{Int}}) == Bool @@ -1142,7 +1142,7 @@ let isa_tfunc = Core.Inference.t_ffunc_val[ end let subtype_tfunc = Core.Inference.t_ffunc_val[ - findfirst(Core.Inference.t_ffunc_key, <:)][3] + findfirst(x->x===(<:), Core.Inference.t_ffunc_key)][3] @test subtype_tfunc(Type{<:Array}, Const(AbstractArray)) === Const(true) @test subtype_tfunc(Type{<:Array}, Type{AbstractArray}) === Const(true) @test subtype_tfunc(Type{<:Array}, Type{AbstractArray{Int}}) == Bool diff --git a/test/libgit2.jl b/test/libgit2.jl index ae809db252d4a5..e24fd220adf413 100644 --- a/test/libgit2.jl +++ b/test/libgit2.jl @@ -50,8 +50,8 @@ end @testset "Check library features" begin f = LibGit2.features() - @test findfirst(f, LibGit2.Consts.FEATURE_SSH) > 0 - @test findfirst(f, LibGit2.Consts.FEATURE_HTTPS) > 0 + @test findfirst(Base.equalto(LibGit2.Consts.FEATURE_SSH), f) > 0 + @test findfirst(Base.equalto(LibGit2.Consts.FEATURE_HTTPS), f) > 0 end @testset "OID" begin diff --git a/test/runtests.jl b/test/runtests.jl index 927165bf4048b4..f83e7b74de8820 100644 --- a/test/runtests.jl +++ b/test/runtests.jl @@ -16,7 +16,7 @@ end const node1_tests = String[] function move_to_node1(t) if t in tests - splice!(tests, findfirst(tests, t)) + splice!(tests, findfirst(Base.equalto(t), tests)) push!(node1_tests, t) end end diff --git a/test/threads.jl b/test/threads.jl index fc0a1d984a84b1..af29f80e742571 100644 --- a/test/threads.jl +++ b/test/threads.jl @@ -25,7 +25,7 @@ function test_threaded_loop_and_atomic_add() @test x[] == 10000 # Next test checks that all loop iterations ran, # and were unique (via pigeon-hole principle). - @test findfirst(found,false) == 0 + @test !(false in found) if was_inorder println(STDERR, "Warning: threaded loop executed in order") end