Haskell-like list comprehension

栏目: IT技术 · 发布时间: 6年前

内容简介:It is a common pattern in functional programming, to the point that some programming languages like Haskell, Clojure, Perl, Python and others do support it directly with syntactic constructs. It allow us to express theLet's take an example:It reflects the

List Comprehension in JS

Install

npm i list-comprehension-in-js

The What

It is a common pattern in functional programming, to the point that some programming languages like Haskell, Clojure, Perl, Python and others do support it directly with syntactic constructs. It allow us to express the set builder-notation into our code.

Let's take an example: S = { 2*x | x € N, x^2 > 100 } , where we are saying "take all the natural number which square is greater than 100 , double them and use these results to create a new set". The resulting set will be: { 22, 24, 26, ... } .

The Why

It reflects the idea of ​​having a set of candidates as solutions to a problem, on which transformations and filtering are applied to narrow the number of them, until right solutions are obtained.

The How

Let's take Haskell as example. The following is the way we can express the previous set comprehension:

[ 2*x | x <- [1..], x^2 > 100 ]

Pretty terse notation, huh? The output will be an infinite, lazy list containing (potentially) all the elements of our set.

Let's take another example. We want to generate all the right triangle that have an even perimeter. This is the Haskell way:

list = [
    {-
        The single element of the resulting list is a triple
        containing the values of the ipothenuse and the two cathets
    -}
    (ipo, cat1, cat2) |

    -- Three ranges for three variables
    ipo <- [1..],
    cat1 <- [1..ipo-1],
    cat2 <-[1..cat1],

    -- Two predicates to satisfy
    ipo^2 == cat1^2 + cat2^2,
    mod (ipo + cat1 + cat2) 2 == 0 ]

Ranges optimizations apart, it's worth noting that:

  • it is possible to have multiple ranges and variables into play
  • it is possible to use the current value taken by a previous defined variable as upper/ lower limit for a following range
  • it is possible to have more predicates to satisfy
  • the output element expression is custom

The How in JavaScript

This library is a custom solution to achieve all of this great possibilities in JS!

It is based on ES6 generators , to achieve the infinite, lazy approach:

const { Comprehension, range } = require("list-comprehension-in-js");

let allRightTrianglesWithEvenPerimeter = new Comprehension(
    // custom output
    (ipo, cat1, cat2) => ({ ipo, cat1, cat2 }),
    [
        // ranges
        () => range(1, Infinity),
        (ipo) => range(1, ipo - 1),
        (_, cat1) => range(1, cat1)
    ],
    [
        // predicates to satisfy
        (ipo, cat1, cat2) => (ipo ** 2 == cat1 ** 2 + cat2 ** 2),
        (ipo, cat1, cat2) => ((ipo + cat1 + cat2) % 2 == 0),
    ]
);

Where range is nothing more than a generator function that provides an iterable of integer numbers and the allRightTrianglesWithEvenPerimeter is itself an iterable too!

So the library perfectly fits in the JavaScript ecosystem.

There is also an utility to extract a finite part of the infinite list, putting the elements into an array:

const { take } = require("list-comprehension-in-js");

const finiteList = take(allRightTrianglesWithEvenPerimeter, 5, 10);

for (const triangle of finiteList) {
    console.log(`hyp: ${triangle.ipo}, c1: ${triangle.cat1}, c2: ${triangle.cat2}`);
}
/*
{hyp: 30, c1: 24, c2: 18}
{hyp: 34, c1: 30, c2: 16}
{hyp: 35, c1: 28, c2: 21}
{hyp: 37, c1: 35, c2: 12}
{hyp: 39, c1: 36, c2: 15}
*/

Utilities API

range

Create an iterable which will output a set of increasing numbers ( start must be <= than end ):

function* range(start, end, step = v => v) {
    // ...
}
  • start : the integer value from which start to generate the numbers
  • end : the last integer value to take
  • step : a mapper function to eventually transform each produced value

rangeReversed

Create an iterable which will output a set of decreasing numbers ( start must be >= than end ):

function* rangeReversed(start, end, step = v => v) {
    // ...
}
  • start : the integer value from which start to generate the numbers
  • end : the last integer value to take
  • step : a mapper function to eventually transform each produced value

take

Take at most nOfElements starting from the index start .

It returns an array containing the elements.

function take(listIterable = [], nOfElements = 0, start = 0) {
    // ...
}
  • listIterable : any iterable, like the one returned by the Comprehension constructor
  • nOfElements : number of elements to pick
  • start : starting index

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

敏捷教练

敏捷教练

[英] Rachel Davies、[英] Liz Sedley / 徐毅、袁店明 / 清华大学出版社 / 2013-7 / 49.00元

《敏捷教练:如何打造优秀的敏捷团队》取材于国际知名敏捷教练的真实经历,展示了他们在辅导团队进行敏捷实践过程中所积累的辅导技巧,凝聚着他们在对敏捷辅导的真知灼见,每章还针对特定主题总结了在转型过程中教练和团队可能面对的障碍及其应对方案。 《敏捷教练:如何打造优秀的敏捷团队》具有较强的实用性和指导性,适合项目经理、技术总监和敏捷团队的所有成员阅读与参考。一起来看看 《敏捷教练》 这本书的介绍吧!

在线进制转换器
在线进制转换器

各进制数互转换器

随机密码生成器
随机密码生成器

多种字符组合密码

Markdown 在线编辑器
Markdown 在线编辑器

Markdown 在线编辑器