Sometimes, it's convenient to combine two lists into a tuple using zip
built-in function in Python. How to make this similarly in Go?
For example:
>>> zip ([1,2],[3,4])
[(1,3), (2,4)]
Sometimes, it's convenient to combine two lists into a tuple using zip
built-in function in Python. How to make this similarly in Go?
For example:
>>> zip ([1,2],[3,4])
[(1,3), (2,4)]
To zip
some number of slice []int
lists,
package main
import "fmt"
func zip(lists ...[]int) func() []int {
zip := make([]int, len(lists))
i := 0
return func() []int {
for j := range lists {
if i >= len(lists[j]) {
return nil
}
zip[j] = lists[j][i]
}
i++
return zip
}
}
func main() {
a := []int{1, 2, 3}
b := []int{4, 5, 6}
c := []int{7, 8, 9, 0}
iter := zip(a, b, c)
for tuple := iter(); tuple != nil; tuple = iter() {
fmt.Println("tuple:", tuple)
}
}
Output:
tuple: [1 4 7] tuple: [2 5 8] tuple: [3 6 9]
If you need the result of the zip
function to be a map
, this can be done with the comparable
constraint
func zip[K comparable, V any](a []K, b []V) map[K]V {
c := make(map[K]V)
for i := 0; i < len(a); i++ {
c[a[i]] = b[i]
}
return c
}
With the support for type parameters, you can write a zip function that zips any two slices.
You can declare a tuple struct that can hold any two types, like this:
type Pair[T, U any] struct {
First T
Second U
}
And the zip function. It can be as simple as:
func Zip[T, U any](ts []T, us []U) []Pair[T,U] {
if len(ts) != len(us) {
panic("slices have different length")
}
pairs := make([]Pair[T,U], len(ts))
for i := 0; i < len(ts); i++ {
pairs[i] = Pair[T,U]{ts[i], us[i]}
}
return pairs
}
Example usage:
func main() {
ts := []uint64{100, 200, 300}
us := []string{"aa", "bb", "cc"}
p := Zip(ts, us)
fmt.Println(p)
// prints [{100 aa} {200 bb} {300 cc}]
}
You can also modify the function above to zip slices of different lengths, by leaving the Pair
field to its zero value for the shorter slice:
func ZipDiff[T, U any](ts []T, us []U) []Pair[T, U] {
// identify the minimum and maximum lengths
lmin, lmax := minmax(len(ts), len(us))
pairs := make([]Pair[T, U], lmax)
// build tuples up to the minimum length
for i := 0; i < lmin; i++ {
pairs[i] = Pair[T, U]{ts[i], us[i]}
}
if lmin == lmax {
return pairs
}
// build tuples with one zero value for [lmin,lmax) range
for i := lmin; i < lmax; i++ {
p := Pair[T, U]{}
if len(ts) == lmax {
p.First = ts[i]
} else {
p.Second = us[i]
}
pairs[i] = p
}
return pairs
}
Example:
func main() {
ts := []uint64{100}
us := []string{"aa", "bb", "cc", "dd", "ee"}
p := ZipDiff(ts, us)
fmt.Println(p)
// prints [{100 aa} {0 bb} {0 cc} {0 dd} {0 ee}]
q := ZipDiff(us, ts)
fmt.Println(q)
// prints [{aa 100} {bb 0} {cc 0} {dd 0} {ee 0}]
}
Code and minmax
helper func available in the playground: https://go.dev/play/p/jpChqsl_GNl
You could do something like this, where you give the tuple type a name:
Or alternatively use an unnamed type for the tuple, like this:
And finally here's a soft-generic way of doing it: