Skip to content

Commit

Permalink
unzip: the inverse of zip
Browse files Browse the repository at this point in the history
  • Loading branch information
StefanKarpinski committed Oct 9, 2019
1 parent e0bdc76 commit c820a50
Show file tree
Hide file tree
Showing 2 changed files with 66 additions and 1 deletion.
1 change: 1 addition & 0 deletions base/exports.jl
Original file line number Diff line number Diff line change
Expand Up @@ -628,6 +628,7 @@ export

enumerate, # re-exported from Iterators
zip,
unzip,
only,

# object identity and equality
Expand Down
66 changes: 65 additions & 1 deletion base/iterators.jl
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,8 @@ import .Base:
getindex, setindex!, get, iterate,
popfirst!, isdone, peek

export enumerate, zip, rest, countfrom, take, drop, cycle, repeated, product, flatten, partition
export enumerate, zip, unzip, rest, countfrom, take, drop,
cycle, repeated, product, flatten, partition

tail_if_any(::Tuple{}) = ()
tail_if_any(x::Tuple) = tail(x)
Expand Down Expand Up @@ -390,6 +391,69 @@ _zip_iterator_eltype(::Type{Tuple{}}) = HasEltype()

reverse(z::Zip) = Zip(map(reverse, z.is))

# unzip

"""
unzip(itrs) -> Vector{<:Vector}
The `unzip` function takes an iterator of iterators and returns a vector of
vectors such that the first vector contains the first element yielded by each
iterator, the second vector the second element yielded by each iterator, etc.
`unzip` is sort of an inverse to the `zip` operation, as the name suggests.
In particular, if we define
≐(a, b) = collect(collect.(a)) == collect(collect.(b))
Then the following identities relating `zip` and `unzip` hold:
unzip(zip(itrs...)) ≐ itrs
zip(unzip(itrs)...) ≐ itrs
Note that `unzip` does not return an iterator: it always consumes all of
its argument and all of each iterator yielded by its argument. It is only
associated with iteration beacuse it is the inverse of `zip` which does
yield an iterator.
# Examples
```jldoctest
julia> unzip(enumerate("Hello"))
2-element Array{Array{T,1} where T,1}:
[1, 2, 3]
['a', 'b', 'c']
julia> unzip([[1, 'a'], [2.5, 'z'], [0, 'x']])
2-element Array{Array{T,1} where T,1}:
Real[1, 2.5, 0]
['a', 'z', 'x']
```
"""
function unzip(itrs)
n = Base.haslength(itrs) ? length(itrs) : -1
vecs = Vector[]
for itr in itrs
for (j, x) in enumerate(itr)
if length(vecs) < j
v = [x]
push!(vecs, v)
n 0 && sizehint!(v, n)
else
v = vecs[j]
if !(x isa eltype(v))
T = Base.promote_typejoin(typeof(x), eltype(v))
v = vecs[j] = copyto!(similar(v, T), v)
n 0 && sizehint!(v, n)
end
push!(v, x)
end
end
length(first(vecs)) == length(last(vecs)) ||
throw(ArgumentError("unzip called with uneven iterators"))
end
return vecs
end

# filter

struct Filter{F,I}
Expand Down

0 comments on commit c820a50

Please sign in to comment.