julia> z = zip([2,4], [6,1]);
julia> findmin(z) do (a,b)
a + b
end
ERROR: MethodError: no method matching keys(::Base.Iterators.Zip{Tuple{Vector{Int64}, Vector{Int64}}})
julia> argmin(enumerate(z)) do (ind, (a,b))
a + b
end
(2, (4, 1))
I understand that findmin returns the index of a value in the domain, whereas argmin returns the value in the domain for which the argument is minimized. However, this sounds like splitting hairs over what the definition of an index is, and whether an index means anything unless you may use it to index into the domain. What I want here is not really an index, but it may be referred to as a count, as I just want to run a counter and return the corresponding value for which the function is minimized. I’m assuming that even though the domain may not be indexed into, one may iterate over it. The argmin variant achieves exactly this. My question is, can findmin be made to behave analogously, or is it reserved for iterators where the domain supports indexing?
That’s the conceptual difference between findmin and argmin: do you want an index in the domain, or a value of the domain? I don’t think that’s splitting hairs. In your case the domain is not indexable so I think it’s neat that you can still do what you want by using argmin with a purpose-built domain that includes the “count”.
Maybe what’s missing from Base is a simple way to wrap an iterator in a type that defines pairs/keys, for example
julia> Base.pairs(it::Iterators.Enumerate) = Iterators.map(Base.splat(=>), it);
julia> z = zip([2,4], [6,1]);
julia> findmin(enumerate(z)) do (a,b)
a + b
end
(5, 2)
I agree it’s not clear cut, but for me the output of enumerate is indeed a mapping (I think the doc here is using plain English, not a formal definition of “mapping” that would require indexing operation to work).
Also it seems hard to argue that Generatorcan have keys but Enumerate should not…
I confess that I have entered an area that I know little about.
But what do you think of the following idea?
I did some tests and it seems to give the expected result in times comparable to the other proposals.
But I’m not sure if that’s a generally valid solution, beyond the example tested.
Any comments are welcome in order to better understand how things are going.
I guess keys don’t make sense for a Zip, as there isn’t a direct key-value relationship. It makes more sense for an Enumerate, although even there the key is a counter, and not something that you may use to index into the parent iterator
Yes I think keys make more sense for Enumerate as the enumeration assigns a kind of ID to each value. And it makes sense that you can wrap any iterator with enumerate to gain this functionality. Otherwise you would have to define methods for all iterators. Or to also cover user iterators you would need to define a generic fallback Base.keys(it), which was turned down here for good reasons I think.
Also for the problem in this thread, defining only keys wouldn’t work for iterators that have no length defined, while enumerating the pairs always works.