iter-tools is an utility toolbox that allows you to unleash the power and expressiveness of iterators and generators.
Create iterators
Create iterators from strings
Transform a single iterable
Combine multiple iterables
Utilities returning multiple iterators
Utilities
Combinatory generators
This should help clarify the documentation. You can also get more informations here: https://developer.mozilla.org/en/docs/Web/JavaScript/Guide/Iterators_and_Generators
- Iterator: an object implementing the iterator protocol (the method next etc.)
- generator: a function returning an iterator
- iterable: a generator function or any object with a generator function under the attribute Symbol.iterator
This package is designed to import only what you need. So if you use one or more iterator you don't need to load the whole library (in the browser for example).
const chain = require('iter-tools/lib/chain');
You can also import the whole library if you find it convenient:
const iterTools = require('iter-tools');
iterTools.chain(iterable1, iterable2);
Every module is available from 2 different folders 'lib' and 'es5'. The latter contains the same iterators but transpile to ES5, ready to be used in older browsers (or old node.js versions).
const chain_es6 = require('iter-tools/lib/chain');
or
const chain_es5 = require('iter-tools/es5/chain');
Create an iterator returning a sequence of numbers (the sequence can be infinite)
const range = require('iter-tools/lib/range');
range(); // 0, 1, 2 ... Infinity
range(3); // 0, 1, 2
range({start: 3}); // 3, 4, 5, 6, 7 ... Infinity
range({start: 3, end: 6}); // 3, 4, 5
range({start: 3, end: 10, step: 3}); // 3, 6, 9
range({start: 9, end: 3, step: -3}); // 9, 6
An alias of range.
const count = require('iter-tools/lib/count');
Create an iterator that returns the same value n times
const repeat = require('iter-tools/lib/repeat');
repeat('x', 3); // 'x', 'x', 'x'
repeat('x'); // 'x', 'x', 'x' .... forever
It cycles the same iterable forever.
const cycle = require('iter-tools/lib/cycle');
cycle(range(3)); // 0, 1, 2, 0, 1, 2, 0, 1, 2 ....
These generators take as a first argument a regular expression and as a second an iterable. If the second argument is omitted it automatically returns a curried function.
It runs a regular expression on a string. Every iteration returns a new match. You should use a "global" regular expression to return multiple matches. The returned object type is the same one returned by the "RegExp.prototype.exec" method (https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/RegExp/exec).
- [0] the full string matching the reg exp
- [1] ... [n] the matching groups
- index: the 0 based index of the match
- input: the original string
const regexpExec = require('iter-tools/lib/regexp-exec');
regexpExec(/[0-9]{4}/g, '10/2/2013, 03/03/2015 12/4/1997');
for (let [match] of iter) {
console.log(match); // '2013', '2015', '1997'
}
Note:
- global regular expressions are mutable, you can't reuse the same object more than once
- the destructuring expression [match] extract only the first match
It splits a string. You can split by regular expression or string.
const regexpSplit = require('iter-tools/lib/regexp-split');
regexpExec(/\s+/g, 'ab s d');
for (let section of iter) {
console.log(section); // ab, s, d
}
Note:
- the regular expression is automatically converted to "global"
- you can use a string (it will be internally transformed to global regExp)
These series of generators take as first argument a function and as a second an iterable. If the second argument is omitted it is automatically returnes a curried function. These functions can be composed:
const compose = require('iter-tools/lib/compose');
const iterator = compose([map(power2), filter(isEven)]);
iterator([ ...... ]);
This is more efficient of using array methods as it doesn't require to build intermediate arrays.
The equivalent of the array "map" function. But runs on an iterable and returns another iterable.
const map = require('iter-tools/lib/map');
map(power2, range(4)); // 0, 1, 4, 9
The equivalent of the array "filter" function. But runs on an iterable and returns another iterable.
const filter = require('iter-tools/lib/filter');
filter(isEven, range(4)); // 0, 2
It returns values as soon as the function is true. Then it stops.
const takeWhile = require('iter-tools/lib/take-while');
takeWhile(isEven, range(4)); // 0
It starts returning values when the function is false. Then it keeps going until the iterator is exausted.
const dropWhile = require('iter-tools/lib/drop-while');
dropWhile(isEven, range(4)); // 1, 2, 3
It returns an iterator that returns a slice of an iterable.
const slice = require('iter-tools/lib/slice');
slice(3, range(10)); // 0, 1, 2
slice({start: 2}, range(10)); // 2, 3, 4, 5, 6, 7, 8, 9
slice({start: 2, end: 6}, range(10)); // 2, 3, 4, 5
slice({start: 2, end: 6, step: 2}, range(10)); // 2, 4
It returns an iterator that returns the progressively reduced value.
const reduceIter = require('iter-tools/lib/reduce-iter');
reduceIter(function (acc, x) {
return acc + x;
}, 0, [0, 1, 2, 3]); // [0, 0], [1, 1], [2, 3], [3, 6]
It maps value of an iterable and flatten them.
const flatMap = require('iter-tools/lib/flat-map');
flatMap(x => [x, x * x], range(4)); // 0, 0, 1, 1, 2, 4, 3, 9
It chains multiple iterables in a single one.
const chain = require('iter-tools/lib/chain');
chain([3, 5, 6], [1, 1], [10]); // 3, 5, 6, 1, 1, 10
It zips 2 or more iterables together. The iteration stops when the shortest iterable is exausted. The first argument is the placeholder used when an iterable is exausted.
const zip = require('iter-tools/lib/zip');
zip([1, 2], [3, 4], [5, 6, 7]); // [1, 3, 5], [2, 4, 6]
It zips 2 or more iterables together. The iteration stops when the longesest iterable is exausted.
const zipLongest = require('iter-tools/lib/zip-longest');
zipLongest(null, [1, 2], [3, 4], [5, 6, 7]); // [1, 3, 5], [2, 4, 6], [null, null, 7]
It is a shorthand for zipping an index to an iterable:
const enumerate = require('iter-tools/lib/enumerate');
enumerate(repeat('x')); // [0, 'x'] [1, 'x'] [2, 'x'] ...
This returns an iterable omitting items when the second iterable, at the same index, contains a falsy value.
const compress = require('iter-tools/lib/compress');
compress(range(5), [0, 0, 1, 1]); // 2, 3
On each iteration it returns a key and a sub-iterator of items with that key. You can pass a function that returns a key, by default an identity function will be used. When you iterate over the next group, the previous sub-iterator items will not be available anymore.
const groupBy = require('iter-tools/lib/groupby');
groupBy([1, 1, 1, 1, -1, -1, -1, 4]);
// It will return:
// 1, subiterator (1, 1, 1, 1)
// -1, subiterator (-1, -1, -1)
// 4, subiterator (4)
groupBy([11, 1, 1, 1, -1, -1, -1, 4], (value) => {value * value});
// It will return:
// 1, subiterator (1, 1, 1, 1, -1, -1, -1)
// 16, subiterator (4)
It returns 2 or more copies of an iterable. In reality they are not copies (it is not possible) they are distinct iterables sharing the original one and caching the values when one of the copy pull a new value from the original iterator.
const tee = require('iter-tools/lib/tee');
tee(range(3)); // [iter1, iter2]
tee(range(3), 4); // [iter1, iter2, iter3, iter4]
This is an implementation of the reduce that consumes an iterable instead of an array. It takes an arguments an iterable, a function and an initial value (default to undefined);
const reduce = require('iter-tools/lib/reduce');
reduce(range(4), (acc, v) => acc += v, 0); // returns 6
It tries to return an iterator from a value. This is useful for 2 reasons:
- you can consume the iterator using the "next" method without worrying if it is a string, array, an iterable etc.
- allows to iterate over a simple object
If the value is an object with a "Symbol.iterator" attribute: it initialise and return the iterator (arrays, maps, sets and strings for example). If the value is already an iterator, it returns itself. If the value is a generator, it initialises it and returns the iterator. If the value is an object, it returns an iterator iterating over attribute/value pairs.
const iter = require('iter-tools/lib/iter');
iter([1, 2, 3]); // 1, 2, 3
iter("hello"); // h e l l o
iter(range(4)); // 0, 1, 2, 3
iter({p1: 1, p2: 2}); // ['p1', 1] ['p2', 2]
It returns an iterator that returns the output of a function at every iteration.
const iter = require('iter-tools/lib/execute');
iter(() => Math.round(Math.random() * 10) ); // 3, 5, 9 ...
This returns the cartesian product of 2 or more iterables. It is equivalent to a nested loop for every iterable.
const product = require('iter-tools/lib/product');
product([1, 2], [3, 4], [5, 6]);
// returns:
// [1, 3, 5],
// [1, 3, 6],
// [1, 4, 5],
// [1, 4, 6],
// [2, 3, 5],
// [2, 3, 6],
// [2, 4, 5],
// [2, 4, 6]
// You can use tee for multiplying the same iterable for itself.
product(...tee(range(2))); // [0, 0] [0, 1] [1, 0] [1, 1]
It returns permutations of length n of an iterable. n defaults to the length of the iterable.
const permutations = require('iter-tools/lib/permutations');
permutations(range(2)); // [0, 1] [1, 0]
permutations([1, 2, 3, 4], 2);
// returns:
// [ 1, 2 ],
// [ 1, 3 ],
// [ 1, 4 ],
// [ 2, 1 ],
// [ 2, 3 ],
// [ 2, 4 ],
// [ 3, 1 ],
// [ 3, 2 ],
// [ 3, 4 ],
// [ 4, 1 ],
// [ 4, 2 ],
// [ 4, 3 ]
It returns combinations of length n of an iterable. n defaults to the length of the iterable.
const combinations = require('iter-tools/lib/combinations');
combinations(range(2)); // [0, 1]
combinations([1, 2, 3, 4], 2);
// returns:
// [ 1, 2 ],
// [ 1, 3 ],
// [ 1, 4 ],
// [ 2, 3 ],
// [ 2, 4 ],
// [ 3, 4 ]
It returns combinations with replacement of length n of an iterable. n defaults to the length of the iterable.
const combinationsWithReplacement = require('iter-tools/lib/combinations-with-replacement');
combinationsWithReplacement(range(2)); // [0, 0] [0, 1] [1, 1]
combinationsWithReplacement([1, 2, 3, 4], 2);
// returns:
// [ 1, 1 ],
// [ 1, 2 ],
// [ 1, 3 ],
// [ 1, 4 ],
// [ 2, 2 ],
// [ 2, 3 ],
// [ 2, 4 ],
// [ 3, 3 ],
// [ 3, 4 ],
// [ 4, 4 ]
There are a couple of limitations that you need to be aware of. First of all, when you consume an iterator object (using next or for..of) you are mutating the object for good. Many of these tools rely on making an in memory copy of the output. For example: cycle, product or tee. They do that in a efficient lazy way. Still you need to consider that. Also with the iterator protocol you can create infinite iterables (repeat, cycle, count etc.). These iterables can't be used by all generators. For example combinatory generators require finite iterables.
Of course I give a lot of credit to the great itertools Python library. It doesn't want to be a mere port, but a properly documented and resonably performant Javascript alternative.