JS高階編程技巧--compose函數

来源:https://www.cnblogs.com/ronaldo9ph/archive/2020/02/17/12323026.html
-Advertisement-
Play Games

先看代碼: let fn1 = function (x) { return x + 10; }; let fn2 = function (x) { return x * 10; }; let fn3 = function (x) { return x / 10; }; console.log(fn3 ...


先看代碼:

let fn1 = function (x) {
    return x + 10;
};
let fn2 = function (x) {
    return x * 10;
};
let fn3 = function (x) {
    return x / 10;
};
  console.log(fn3(fn1(fn2(fn1(6)))));

這是幾個簡單的運算方法,但想輸出的是一個多層函數嵌套的運行結果,即把前一個函數的運行結果賦值給後一個函數,當然我們可以寫成一下這樣:

let x = fn1(6);
x = fn2(x);
x = fn1(x);
x = fn3(x);

 

但現在我就想用一個函數解決這種問題,形如:

compose(fn1, fn2, fn1, fn3)(6);

 

這個compose函數就是這篇文章介紹的——函數調用的扁平化,即把層級嵌套的那種函數調用(一個函數的運行結果當作實參傳給下一個函數的這種操作)扁平化,這就是compose函數。

 

那麼下麵就是開始實現這個函數:

首先我們看參數,需要給出不確定個數的函數:

function compose(...funcs) {
    //=>funcs:傳遞的函數集合
}

 

compose函數執行後跟個(),說明函數執行完再執行一個函數,即函數執行完會返回一個新函數,而且也會給出第一次調用函數時的參數:

function compose(...funcs) {
    //=>funcs:傳遞的函數集合
    return function proxy(...args) { 
      //=>args:第一次調用函數傳遞的參數集合

    }    
}

 

繼續往下進行,我們需要判斷給出的函數集合的個數,如果沒有給函數,我們只需將後一個的參數返回,如果只給出一個函數,我們只需把後一個的參數賦給這個函數去執行即可:

function compose(...funcs) {
    //=>funcs:傳遞的函數集合
    return function proxy(...args) {
        //=>args:第一次調用函數傳遞的參數集合
        let len = funcs.length;
        if (len === 0) {
            //=>一個函數都不需要執行,直接返回ARGS
            return args;
        }
        if (len === 1) {
            //=>只需要執行一個函數,把函數執行,把其結果返回即可
            return funcs[0](...args);
        }

    };
}

 

如果給出的參數集合是兩個及以上,那就是把前一個函數的執行結果賦給後一個函數,說到這,應該會想到一個滿足這個需求的數組方法——reduce:

function compose(...funcs) {
    //=>funcs:傳遞的函數集合
    return function proxy(...args) {
        //=>args:第一次調用函數傳遞的參數集合
        let len = funcs.length;
        if (len === 0) {
            //=>一個函數都不需要執行,直接返回ARGS
            return args;
        }
        if (len === 1) {
            //=>只需要執行一個函數,把函數執行,把其結果返回即可
            return funcs[0](...args);
        }
        return funcs.reduce((x, y) => {
        
        });
    };
}

 

但這裡需要註意的是,第一次執行的時候,參數x是個函數,之後再執行的時候x是個函數執行的結果,所以需要進行判斷:

function compose(...funcs) {
    //=>funcs:傳遞的函數集合
    return function proxy(...args) {
        //=>args:第一次調用函數傳遞的參數集合
        let len = funcs.length;
        if (len === 0) {
            //=>一個函數都不需要執行,直接返回ARGS
            return args;
        }
        if (len === 1) {
            //=>只需要執行一個函數,把函數執行,把其結果返回即可
            return funcs[0](...args);
        }
        return funcs.reduce((x, y) => {
            return typeof x === "function" ? y(x(...args)) : y(x)
        });
    };
}

 

這樣,compose函數完成。

 

當然,redux源碼中的compose.js也可以實現一開始想要的效果:

export default function compose(...funcs) {
  if (funcs.length === 0) {
    return arg => arg
  }

  if (funcs.length === 1) {
    return funcs[0]
  }

  return funcs.reduce((a, b) => (...args) => a(b(...args)))
}

 

但它和我們寫的compose函數有些不同,它執行的順序是函數集合中的函數從後往前執行,所以結果也會不同:

compose(fn1, fn2, fn1, fn3)(6)); //=> 用第一個compose執行的結果是17,用redux的執行結果是116

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • HDFS(Hadoop Distributed File System) 分散式文件系統,HDFS是一個高度容錯性的系統,適合部署在廉價的機器上。HDFS能提供高吞吐量的數據訪問,非常適合大規模數據集上的應用.由NameNode,若幹DataNode,以及Secondary NameNode組成。 ...
  • Hadoop是一個由Apache基金會所開發的分散式基礎架構,Hadoop的框架最核心的設計就是:HDFS和MapReduce。HDFS為海量的數據提供了存儲,而MapReduce則為海量的數據提供了計算,特點是:高可靠性,高擴展性,高效性,高容錯性。 Hadoop與Google三篇論文 Googl ...
  • 不久前自學完完sql,下了mysql8.0.17,安裝配置好後探索著,想著用root賬戶登上去能不能刪除root賬戶呢,然後就想給自己一巴掌,,, 如何快速恢復root: 1.關閉mysql服務:win+R鍵鍵入services.msc,找到mysql服務,點擊stop; 2.刪除data文件夾及其 ...
  • 在Druid快速入門其實已經簡單的介紹過最簡化配置的單節點部署,本文我們將詳細描述Druid的多種部署方式,對於測試開發環境可以選用輕量的單機部署方式,而生產環境我們最好選用集群部署的方式,確保系統的高可用性。 一、單機部署 Druid提供了一組可以參考的配置和單機部署的啟動腳本。 尺寸適合筆記本電 ...
  • 定義 對象是JS中的引用數據類型。對象是一種複合數據類型,在對象中可以保存多個不同數據類型的屬性。使用typeof檢查一個對象時,會返回object。 分類 內置對象 由ES標准定義的對象,在任何ES的實現中都可以實現。比如 Math String Number Boolean Function O ...
  • BOM(Browser Object Model)也叫瀏覽器對象,它提供了很多對象,用於訪問瀏覽器的功能。但是BOM是沒有標準的,每一個瀏覽器廠家會根據自己的需求來擴展BOM對象。本文主要以一些簡單的小例子,簡述前端開發中BOM的相關基礎知識,僅供學習分享使用,如有不足之處,還請指正。 ...
  • ES6)新增加了兩個重要的 JavaScript 關鍵字:let 和 const。以前聲明變數時只有一種方式:var,ES6對聲明方式進行了擴展,現在可以有三種聲明方式了。 1、var:variable的簡寫,字面意思就是變數。 2、let:let的意思(vt. 允許,讓;出租;假設;妨礙;vi. ...
  • 動畫DOM及CSS操作 自定義動畫 animate(最終css狀態,時間) 這個最終css狀態是一個對象 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>demo</title> <style> div{ w ...
一周排行
    -Advertisement-
    Play Games
  • Dapr Outbox 是1.12中的功能。 本文只介紹Dapr Outbox 執行流程,Dapr Outbox基本用法請閱讀官方文檔 。本文中appID=order-processor,topic=orders 本文前提知識:熟悉Dapr狀態管理、Dapr發佈訂閱和Outbox 模式。 Outbo ...
  • 引言 在前幾章我們深度講解了單元測試和集成測試的基礎知識,這一章我們來講解一下代碼覆蓋率,代碼覆蓋率是單元測試運行的度量值,覆蓋率通常以百分比表示,用於衡量代碼被測試覆蓋的程度,幫助開發人員評估測試用例的質量和代碼的健壯性。常見的覆蓋率包括語句覆蓋率(Line Coverage)、分支覆蓋率(Bra ...
  • 前言 本文介紹瞭如何使用S7.NET庫實現對西門子PLC DB塊數據的讀寫,記錄了使用電腦模擬,模擬PLC,自至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1.Windows環境下鏈路層網路訪問的行業標準工具(WinPcap_4_1_3.exe)下載鏈接:http ...
  • 從依賴倒置原則(Dependency Inversion Principle, DIP)到控制反轉(Inversion of Control, IoC)再到依賴註入(Dependency Injection, DI)的演進過程,我們可以理解為一種逐步抽象和解耦的設計思想。這種思想在C#等面向對象的編 ...
  • 關於Python中的私有屬性和私有方法 Python對於類的成員沒有嚴格的訪問控制限制,這與其他面相對對象語言有區別。關於私有屬性和私有方法,有如下要點: 1、通常我們約定,兩個下劃線開頭的屬性是私有的(private)。其他為公共的(public); 2、類內部可以訪問私有屬性(方法); 3、類外 ...
  • C++ 訪問說明符 訪問說明符是 C++ 中控制類成員(屬性和方法)可訪問性的關鍵字。它們用於封裝類數據並保護其免受意外修改或濫用。 三種訪問說明符: public:允許從類外部的任何地方訪問成員。 private:僅允許在類內部訪問成員。 protected:允許在類內部及其派生類中訪問成員。 示 ...
  • 寫這個隨筆說一下C++的static_cast和dynamic_cast用在子類與父類的指針轉換時的一些事宜。首先,【static_cast,dynamic_cast】【父類指針,子類指針】,兩兩一組,共有4種組合:用 static_cast 父類轉子類、用 static_cast 子類轉父類、使用 ...
  • /******************************************************************************************************** * * * 設計雙向鏈表的介面 * * * * Copyright (c) 2023-2 ...
  • 相信接觸過spring做開發的小伙伴們一定使用過@ComponentScan註解 @ComponentScan("com.wangm.lifecycle") public class AppConfig { } @ComponentScan指定basePackage,將包下的類按照一定規則註冊成Be ...
  • 操作系統 :CentOS 7.6_x64 opensips版本: 2.4.9 python版本:2.7.5 python作為腳本語言,使用起來很方便,查了下opensips的文檔,支持使用python腳本寫邏輯代碼。今天整理下CentOS7環境下opensips2.4.9的python模塊筆記及使用 ...