From 9c6f31c93c96e35acbccf20ade665c312d7d0b2e Mon Sep 17 00:00:00 2001 From: Patrick Henning Date: Wed, 6 Oct 2021 15:10:53 -0700 Subject: [PATCH] Add zip and unzip functions --- src/arrays.ts | 52 +++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 52 insertions(+) diff --git a/src/arrays.ts b/src/arrays.ts index c6b0b6a..73393b5 100644 --- a/src/arrays.ts +++ b/src/arrays.ts @@ -158,6 +158,58 @@ export function join(...arrays: T[][]): T[] { return arrays.reduce((a, x) => a.concat(x), []); } +/** Turn a 2-tuple of lists into a list of 2-tuples. For example: `[['a', 'b', 'c'], [1, 2, 3]]` becomes `[['a', 1], ['b', 2], ['c', 3]]` */ +export function zip2(lists: [A[], B[]]) { + return zipAny(lists) as Array<[A, B]>; +} + +/** Turn a 3-tuple of lists into a list of 3-tuples. */ +export function zip3(lists: [A[], B[], C[]]) { + return zipAny(lists) as Array<[A, B, C]>; +} + +/** Turn a 4-tuple of lists into a list of 4-tuples. */ +export function zip4(lists: [A[], B[], C[], D[]]) { + return zipAny(lists) as Array<[A, B, C, D]>; +} + +/** Turn a n-tuple of lists into a list of n-tuples. */ +export function zipAny(lists: any[][]) { + const zipped = lists[0].map(() => []) as any[][]; + for (const [index, zippedList] of zipped.entries()) { + for (const list of lists) { + zippedList.push(list[index]); + } + } + return zipped; +} + +/** Turn a list of 2-tuples into a 2-tuple of lists. For example: `[['a', 1], ['b', 2], ['c', 3]]` becomes `[['a', 'b', 'c'], [1, 2, 3]]` */ +export function unzip2(tuples: Array<[A, B]>) { + return unzipAny(tuples) as [A[], B[]]; +} + +/** Turn a list of 3-tuples into a 3-tuple of lists. */ +export function unzip3(tuples: Array<[A, B, C]>) { + return unzipAny(tuples) as [A[], B[], C[]]; +} + +/** Turn a list of 4-tuples into a 4-tuple of lists. */ +function unzip4(tuples: Array<[A, B, C, D]>) { + return unzipAny(tuples) as [A[], B[], C[], D[]]; +} + +/** Turn a list of n-tuples into a n-tuple of lists. */ +function unzipAny(tuples: any[][]) { + const init = tuples.map(() => []) as any[][]; + return tuples.reduce( + (unzipped: any[][], tuple) => { + return tuple.map((elem, index) => [...unzipped[index], elem]); + }, + init + ); +} + type LinkedListItem = {val: T, prev: LinkedListItem, next: LinkedListItem}; /** Converts an array to a linked list data structure. */