Detailed explanation of the usage of compose function and pipe function in JS

Detailed explanation of the usage of compose function and pipe function in JS

compose function

The compose function can flatten functions that need to be executed in nested order. Nested execution means that the return value of one function will be used as the parameter of another function. Let's consider a simple requirement: This requirement is very simple, and a calculation function will do:

const calculate = x => (x + 10) * 10;
let res = calculate(10);
console.log(res); // 200

But according to the functional programming we talked about before, we can break down the complex steps into several simple and reusable steps, so we broke out an addition function and a multiplication function:

const add = x => x + 10;
const multiply = x => x * 10;

// Our calculation is changed to a nested calculation of two functions, and the return value of the add function is used as the parameter of the multiply function let res = multiply(add(10));
console.log(res); // The result is still 200

The calculation method above is the nested execution of functions, and the role of our compose is to flatten the nested execution methods as parameters. When nested execution occurs, the method inside, that is, the method on the right, is executed first, and then returns to the left. Our compose method also starts from the parameter on the right, so our goal is very clear. We need a compose method like this:

// Parameters are executed from right to left, so multiply comes first and add comes last let res = compose(multiply, add)(10);

Before we talk about this, let's take a look at a function we need to use: Array.prototype.reduce

Array.prototype.reduce

The reduce method of an array can achieve an accumulation effect. It receives two parameters, the first is an accumulator method, and the second is the initialization value. The accumulator receives four parameters. The first one is the last calculated value, and the second one is the current value of the array. These two parameters are mainly used. The last two parameters are not commonly used. They are the current index and the array of the current iteration:

const arr = [[1, 2], [3, 4], [5, 6]];
// The initial value of prevRes is the passed [], and it will be the value calculated after each iteration const flatArr = arr.reduce((prevRes, item) => prevRes.concat(item), []);

console.log(flatArr); // [1, 2, 3, 4, 5, 6]

Array.prototype.reduceRight

Array.prototype.reduce will iterate from left to right. If you need to iterate from right to left, use Array.prototype.reduceRight.

const arr = [[1, 2], [3, 4], [5, 6]];
// The initial value of prevRes is the passed [], and it will be the value calculated after each iteration const flatArr = arr.reduceRight((prevRes, item) => prevRes.concat(item), []);

console.log(flatArr); // [5, 6, 3, 4, 1, 2]

So how do we implement this compose method? We need to use Array.prototype.reduceRight:

const compose = function(){
  // Store the received parameters into an array, args == [multiply, add]
  const args = [].slice.apply(arguments);
  return function(x) {
    return args.reduceRight((res, cb) => cb(res), x);
  }
}

// Let's verify this method let calculate = compose(multiply, add);
let res = calculate(10);
console.log(res); // The result is still 200

The compose function above would be more concise using ES6:

const compose = (...args) => x => args.reduceRight((res, cb) => cb(res), x);

Redux's middleware is implemented using compose, and the loading order of loader in webpack is also from right to left, because it is also implemented using compose.

pipe Function

The pipe function is the same as the compose function, and the parameters are also laid out flatly, but the order is from left to right. Let's implement it. Just change reduceRight to reduce:

const pipe = function(){
  const args = [].slice.apply(arguments);
  return function(x) {
    return args.reduce((res, cb) => cb(res), x);
  }
}

// Change the order of parameters to from left to right let calculate = pipe(add, multiply);
let res = calculate(10);
console.log(res); // The result is still 200

ES6 writing:

const pipe = (...args) => x => args.reduce((res, cb) => cb(res), x)

The above is a detailed explanation of the usage of the compose function and pipe function in JS. For more information about JS, please pay attention to other related articles on 123WORDPRESS.COM!

You may also be interested in:
  • Node.js pipe implements source code parsing
  • Understand the principles and implementation of nodejs's stream and pipe mechanisms
  • Detailed explanation of JavaScript Reduce
  • Detailed explanation of JavaScript Array.reduce source code
  • 5 Basic Usage Examples of reduce() in JavaScript
  • Analysis of the principle and usage skills of JS array reduce() method
  • Detailed explanation of the function and usage of JS array Reduce method
  • Example analysis of the usage of JS array method reduce
  • JavaScript reduce and reduceRight explained

<<:  Solve the problem that VMWare cannot display in full screen after installing Mac system

>>:  8 examples of using killall command to terminate processes in Linux

Recommend

CSS scroll-snap scroll event stop and element position detection implementation

1. Scroll Snap is a must-have skill for front-end...

MySQL database operations and data types

Table of contents 1. Database Operation 1.1 Displ...

How to start and stop SpringBoot jar program deployment shell script in Linux

Without further ado, let me give you the code. Th...

How to fill items in columns in CSS Grid Layout

Suppose we have n items and we have to sort these...

Detailed explanation of MySQL deadlock and database and table sharding issues

Record the problem points of MySQL production. Bu...

mysql 5.6.23 winx64.zip installation detailed tutorial

For detailed documentation on installing the comp...

JS 4 super practical tips to improve development efficiency

Table of contents 1. Short circuit judgment 2. Op...

Using JS to implement a rotating Christmas tree in HTML

<!DOCTYPE HEML PUBLIC> <html> <hea...

How to convert a string into a number in JavaScript

Table of contents 1.parseInt(string, radix) 2. Nu...

How to use Docker to build a tomcat cluster using nginx (with pictures and text)

First, create a tomcat folder. To facilitate the ...

How to use custom images in Html to display checkboxes

If you need to use an image to implement the use ...

How to avoid data loop conflicts when MySQL is configured with dual masters

I wonder if you have ever thought about this ques...