Emulating Python's zip() and zip_longest() in JavaScript
This is essentially how you emulate the way Python's zip()
function works using JavaScript.
1let a = [1, 3, 5];
2let b = [2, 4, 6];
3const mapping = a.map((item, idx) => [item, b[idx]]);
4console.log(mapping);
5// result
6[ [ 1, 2 ], [ 3, 4 ], [ 5, 6 ] ]
What makes this possible is the fact that JavaScript array methods — map()
, forEach()
and even filter()
— take up to three arguments (element, index, array) while they iterate through the array.
In which element
is the element that the iterator points to at each step, index
is the index of this item in the original array. And it's worth noting the third argument is the original array.
Since map()
method of the JavaScript array is more flexible than zip()
, we can do more stuff to the iterated item or the array. But if you want to do anything with array
, remember the original array changes accordingly. In this case, a
:
1let a = [1, 3, 5];
2let b = [2, 4, 6];
3const mapping = a.map((item, idx, array) => {
4 array.push(item*2);
5 return [item, b[idx]]} );
6console.log(mapping);
7// Result
8[ [ 1, 2 ], [ 3, 4 ], [ 5, 6 ] ]
9console.log(a);
10// Result
11[ 1, 3, 5, 2, 6, 10 ]
As we can see, integers 2, 6 and 10 are pushed to the end of the array.
Python provides a zip_longest()
function in itertools
.
While zip()
cuts off longer lists to fit the shortest, zip_longest()
stretches shorter lists to match the longest. Empty slots in shorter lists are filled with a given value. By default this fillvalue
is None
.
This is how this function works:
1import itertools
2
3a = [1, 3, 5, 7, 9]
4b = [-1, -3, -5]
5c = itertools.zip_longest(a, b)
6
7print(list(c))
8# Result
9[(1, -1), (3, -3), (5, -5), (7, None), (9, None)]
Below I try to emulate this function in JavaScript.
First let's take a look at line 👀. Notably, In order to take indefinite number of arguments, I'm using the ellipsis operator. Unlike unpacking a list using the star operator (*) in Python, using something like ...args, fillvalue=null
will raise an error. So I have to use a pair of square brackets to allow for a fillvalue
argument. (Let's give it a duck.)
1let a = [5, 3, 1, -1, -3, -5]
2let b = [6, 4, 2, 0, -2, -4, -6, -8]
3let c = ['a', 'b', 'c', 'd']
4
5const zipLongest = ([...args], fillvalue=null) => { 👀
6 const result = []
7 let i=0
8 while (args.some( argArray=> argArray[i])) {
9 const ithColumn = args.map(
10 argArray=> {
11 item = (typeof argArray[i] === 'undefined') 👓
12 ? fillvalue
13 : argArray[i]
14 return item
15 }
16 )
17 result.push(ithColumn)
18 i++
19 }
20 return result
21}
22
23console.log(zipLongest([a, b, c], fillvalue="duck"))
24
25// Result
26[
27 [ 5, 6, 'a' ],
28 [ 3, 4, 'b' ],
29 [ 1, 2, 'c' ],
30 [ -1, 0, 'd' ], 🙄
31 [ -3, -2, 'duck' ],
32 [ -5, -4, 'duck' ],
33 [ 'duck', -6, 'duck' ],
34 [ 'duck', -8, 'duck' ]
35]
Then let's move to line 👓. Empty slots in the JavaScript array are considered undefined
.
But undefined
and other falsy values, eg empty strings and 0, all evaluates to false
. This will confused this ternary conditional. As a result we will get [ -1, undefined, 'd' ]
on line 🙄.
A better way is to check the type of item as we iterate. However, a wrong way to do it is using typeof nonExistentValue === undefined
.
Although the nonExistentValue
is indeed not defined, typeof
returns "undefined"
, which is a string. And this string is a string. It is by no means equal to the reserved word for non-existent things. Thus the condition we used on line 👓.