-
-
Notifications
You must be signed in to change notification settings - Fork 5.6k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Allow map
to work on Iterators.Pairs
#38150
base: master
Are you sure you want to change the base?
Conversation
Counter-argument: if we are serious about |
So I guess the concern (in brief) is that if
Another option would be to make julia> Base.collect(f, itr) = collect(Base.Generator(f, itr));
julia> Base.collect(f, itrs...) = collect(Base.Generator(Base.splat(f), Iterators.product(itrs...)));
julia> map(enumerate(rand(Bool, 2,10))) do (i,v)
i * v
end
2×10 Array{Int64,2}:
0 0 0 7 0 11 13 15 17 19
2 4 6 0 10 0 14 0 0 20
julia> collect(pairs(rand(Bool, 2,10))) do (i,v)
prod(i.I) * v
end
2×10 Array{Int64,2}:
1 0 3 4 5 6 0 0 0 10
2 0 0 8 0 0 0 0 0 0 Which might be nice in any case, as a convenient way to write outer products with julia> collect(1:2, 4:10) do x,y
string(x, '/', y)
end
2×7 Matrix{String}:
"1/4" "1/5" "1/6" "1/7" "1/8" "1/9" "1/10"
"2/4" "2/5" "2/6" "2/7" "2/8" "2/9" "2/10"
julia> ans == [string(x, '/', y) for x in 1:2, y in 4:10]
true |
I've always been skeptical of whether Iterators.Pairs is really a dictionary. The only thing it's supposed to be is an iterator over index-value pairs of some collection. |
Oh right, we actually have a |
For example, |
Right, I think that example shows why |
base/iterators.jl
Outdated
@@ -272,6 +272,8 @@ setindex!(v::Pairs, value, key) = (v.data[key] = value; v) | |||
get(v::Pairs, key, default) = get(v.data, key, default) | |||
get(f::Base.Callable, v::Pairs, key) = get(f, v.data, key) | |||
|
|||
Base.map(f, A::Pairs) = collect(Base.Generator(f,A)) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Perhaps should be general (and in abstractdict.jl)?
Base.map(f, A::Pairs) = collect(Base.Generator(f,A)) | |
Base.map(f, A::AbstractDict) = collect(Base.Generator(f, A)) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
triage says yes.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
To be clear, this means that map(f, d::Dict)
would return a vector? Then it's the same as map(f, d)
and we have no way to apply a function and get a dictionary back like in Dictionaries.jl?
Really looking forward for I believe it makes sense to define However, would be ideal to have |
@@ -322,6 +322,8 @@ setindex!(v::Pairs, value, key) = (getfield(v, :data)[key] = value; v) | |||
get(v::Pairs, key, default) = get(getfield(v, :data), key, default) | |||
get(f::Base.Callable, v::Pairs, key) = get(f, getfield(v, :data), key) | |||
|
|||
Base.map(f, v::Pairs) = Base.collect_similar(values(v), Base.generator(f, v)) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This version handles unusual arrays better, probably needs more tests:
julia> using StaticArrays
julia> z = SA[10, 20, 30];
julia> map(pairs(z)) do (ind, val)
val + ind
end
3-element MVector{3, Int64} with indices SOneTo(3):
11
22
33
julia> using StructArrays
julia> x = StructArray([(sqr=i^2, str=string(i)) for i in 1:3]);
julia> map(pairs(x)) do (ind, nt)
(; ind, nt)
end
3-element StructArray(::Vector{Int64}, ::Vector{NamedTuple{(:sqr, :str), Tuple{Int64, String}}}) with eltype NamedTuple{(:ind, :nt), Tuple{Int64, NamedTuple{(:sqr, :str), Tuple{Int64, String}}}}:
(ind = 1, nt = (sqr = 1, str = "1"))
(ind = 2, nt = (sqr = 4, str = "2"))
(ind = 3, nt = (sqr = 9, str = "3"))
With just collect
not collect_similar
(as in first commit), a StaticArray lead to a SizedVector, and a StructArray leads to a Vector. I believe that offsets should work with either version:
julia> using OffsetArrays;
julia> y = OffsetArray([10, 20, 30], 4);
julia> map(pairs(y)) do (ind, val)
val + ind
end
3-element OffsetArray(::Vector{Int64}, 5:7) with eltype Int64 with indices 5:7:
15
26
37
There are many counter examples to this, since |
Regular So, I don't really understand why
Regular |
Yes, it seems nice to try to call I'm not sure why Pairs is an AbstractDict, maybe an AbstractArray would have been better. Probably that can't be changed now. We could introduce a non-dictionary Note BTW that it does already work in
Pairs also doesn't support broadcasting. It could be fixed with something like this:
|
From the comments above, it looks like there is a total consensus that |
Bump, for 1.9 perhaps? Tests probably need elaborating for the |
A suggestion that may make the PR more focused, less controversial, and potentially lead to even more generic code in the future: to make Thoughts? |
Bump for 1.12? It doesn't seem that controversial above. So it's not clear to me that I was wondering if this ought to be documented, and where, but it's a bit awkward. The function Since help?> pairs
search: pairs Pair parse popfirst
pairs(collection)
Return an iterator over key => value pairs for any collection that maps a set of keys to a set
of values. This includes arrays, where the keys are the array indices.
[...]
help?> Base.Pairs
│ Warning
│
│ The following bindings may be internal; they may change or be removed in future
│ versions:
│
│ • Base.Pairs
Base.Pairs(values, keys) <: AbstractDict{eltype(keys), eltype(values)}
Transforms an indexable container into a Dictionary-view of the same data. Modifying the
key-space of the underlying data may invalidate this object.
|
map
works on many iterators, but not on pairs, becauseIterators.Pairs <: AbstractDict
. Is there a good reason to prevent this?Stumbled across this here: https://discourse.julialang.org/t/enumerate-multi-dimensional-array/48877/7
Edit: earlier discussion at #5794, about
Dict
as well aspair
.