亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

使用 JavaScript 進行函數式編程 (一) 翻譯

 更新時間:2015年10月02日 13:57:40   作者:simei231, 祝青, 李中凱  
本文是函數式編程系列的第一篇文章。這里我會簡要介紹一下編程范式,然后會直接介紹使用 Javascript 進行函數式編程的概念,因為 JavsScript 是最被認可的函數式程序語言之一。我們鼓勵讀者通過參考資料部分進一步了解這一迷人的概念

編程范式

編程范式是一個由思考問題以及實現問題愿景的工具組成的框架。很多現代語言都是聚范式(或者說多重范式): 他們支持很多不同的編程范式,比如面向對象,元程序設計,泛函,面向過程,等等。

Bird

函數式編程范式

函數式編程就像一輛氫燃料驅動的汽車——先進的未來派,但是還沒有被廣泛推廣。與命令式編程相反,他由一系列語句組成,這些語句用于更新執(zhí)行時的全局狀態(tài)。函數式編程將計算轉化作表達式求值。這些表達式全由純數學函數組成,這些數學函數都是一流的(可以被當做一般值來運用和處理),并且沒有副作用。

Programming Paradigm

函數式編程很重視以下值:

函數是一等要務

我們應該將函數與編程語言中的其他類對象同樣對待。換句話說,您可以將函數存儲在變量里,動態(tài)創(chuàng)建函數,以及將函數返回或者將函數傳遞給其他函數。下面我們來看一個例子...

Flight Attendant

一個字符串可以保存為一個變量,函數也可以,例如:

var sayHello = function() { return “Hello” };

一個字符串可以保存為對象字段,函數也可以,例如:

var person = {message: “Hello”, sayHello: function() { return “Hello” }};

一個字符串可以再用到時才創(chuàng)建,函數也可以,例如:

“Hello ” + (function() { return “World” })(); //=> Hello World

一個字符串可以作為輸入參數傳給函數,則函數也可以:

    function hellloWorld(hello, world) { return hello + world() }

一個字符串可以作為函數返回值,函數也可以,例如:

return “Hello”;
return function() { return “Hello”};

高階案例

Angels

如果函數將其他函數函數作為輸入參數或者作為返回值,則稱之為高階函數。剛才我們已經看過了一個高階函數的例子。下面,我們來看一下更復雜的情況。

例1

[1, 2, 3].forEach(alert);
// alert 彈窗顯示“1" 
// alert 彈窗顯示 "2" 
// alert 彈窗顯示 "3”

例2

function splat(fun) {
   return function(array) {
        return fun.apply(null, array);
   };
}
var addArrayElements = splat(function(x, y) { return x + y });
addArrayElements([1, 2]);
//=> 3

最愛純函數

Pure Functions


純函數不會有其他的副作用,所謂的副作用指的是函數所產生的對函數外界狀態(tài)的修改。比如:

  • 修改某個變量

  • 修改數據結構

  • 對外界某個變量設置字段

  • 拋出例外或者彈出錯誤信息

最簡單的例子就是數學函數。Math.sqrt(4) 函數總是返回2。他不會用到任何其他心寒信息,如狀態(tài)或者設置參數。數學函數從來不會造成任何副作用。


避免修改狀態(tài)


Pure functions cannot mutate data

函數式編程支持純粹的函數,這樣的函數不能改變數據,因此大多用于創(chuàng)建不可改變的的數據。這種方式,不用修改一個已存在的數據結構,而且能高效的新建一個.
你也許想知道,如果一個純粹的函數通過改變一些本地數據而生產一個不可改變的返回值,是否是允許的?答案是可以。
在JavaScript中極少的數據類型是默認是不可改變的。String是一個不能被改變的數據類型的例子:

   var s = "HelloWorld";
    s.toUpperCase();
    //=> "HELLOWORLD"
    s;
    //=> "HelloWorld"

不可改變狀態(tài)的好處

    •    避免混亂和增加程序的準確性:在復雜系統(tǒng)內,大多數難以理解的Bug是由于狀態(tài)通過在程序中外部客戶端代碼修改而導致的。

    •    確立“快速簡潔”的多線程編程:如果多線程可以修改同一個共享值,你不得不同步的獲取值。這對專家來說都是十分乏味并且易出錯的編程挑戰(zhàn)。

軟件事務內存和Actor模型提供了直接在線程安全方式下處理修改。

使用遞歸而非循環(huán)調用

Recursion

遞歸是最有名的函數式編程技術。如果您還不知道它的話,那么可以理解為遞歸函數就是一個可以調用自己的函數。

替代反復循環(huán)的最經典方式就是使用遞歸,即每次完成函數體操作之后,再繼續(xù)執(zhí)行集合里的下一項,直到滿足結束條件。遞歸還天生符合某些算法實現,比如遍歷樹形結構(每個樹枝都是一顆小樹)。

在任何語言里,遞歸都是一項重要的函數式編程方式。很多函數語言甚至要求的更加嚴格:只支持遞歸遍歷,而不支持顯式的循環(huán)遍歷。這需要語言必須保證消除了尾端調用,這是 JavasSrip 不支持的。

惰性求值優(yōu)于激進計算

Laziness is not always bad!

數學定義了很多無窮集合,比如自然數(所有的正整數)。他們都是符號表示。任意特定有限的子集都在需要時求值。我們將其稱之為惰性求值(也叫做非嚴格求值,或者按需調用,延遲執(zhí)行)。及早求值會強迫我們表示出所有無窮數據,而這顯然是不可能的。

很多語言都默認是惰性的,有些也提供了惰性數據結構以表達無窮集合,并在需要時對自己進行精確計算。

很明顯一行代碼 result = compute() 所表達的是將 compute() 的返回結果賦值給 result。但是 result 的值究竟是多少只有其被用到的時候才有意義。

可見策略的選擇會在很大程度上提高性能,特別是當用在鏈式處理或者數組處理的時候。這些都是函數式程序員所喜愛的編程技術。

這就開創(chuàng)可很多可能性,包括并發(fā)執(zhí)行,并行技術以及合成。

但是,有一個問題,JavaScrip 并不對自身進行惰性求值。話雖如此,Javascript 里的函數庫可以有效地模擬惰性求值。

閉包的全部好處

所有的函數式語言都有閉包,然而這個語言特性經常被討論得很神秘。閉包是一個函數,這個函數有著對內部引用的所有變量的隱式綁定。換句話說,該函數對它引用的變量封閉了一個上下文。JavaScript 中的閉包是能夠訪問父級作用域的函數,即使父級函數已經調用完畢。

   function multiplier(factor) {
      return function(number) {
          return number * factor;
      };
   }
  var twiceOf = multiplier(2);
    console.log(twiceOf(6));
//=> 12

聲明式優(yōu)于命令式編程

函數式編程是聲明式的,就像數學運算,屬性和關系是定義好的。運行時知道怎么計算最終結果。階乘函數的定義提供了一個例子:

factorial(n)       = 1 if n = 1

                            n * factorial(n-1) if n > 1

該定義將 factorial(n) 的值關聯到 factorial(n-1),是遞歸定義。特殊情況下的 factorial(1) 終止了遞歸。

var imperativeFactorial = function(n) {
    if(n == 1) {
        return 1
    } else {
        product = 1;
        for(i = 1; i <= n; i++) {
              product *= i;
        }
        return product;
     }
}
var declarativeFactorial = function(n) {
       if(n == 1) {
             return 1
       } else {
             return n * factorial(n - 1);
      }
  }

從它實現階乘計算來看,聲明式的階乘可能看起來像“命令式”的,但它的結構更像聲明式的。

命令式階乘使用可變值、循環(huán)計數器和結果來累加計算后的結果。這個方法顯式地實現了特定的算法。不像聲明式版本,這種方法有許多可變步驟,導致它更難理解,也更難避免 bug 。

Imperative vs Functional Approach

函數式JavaScript庫

有很多函數式庫:underscore.js, lodash,Fantasy Land, Functional.js, Bilby.js, fn.js, Wu.js, Lazy.js, Bacon.js, sloth.js, stream.js, Sugar, Folktale, RxJs 等等。

函數式程序員工具包

map(), filter(), 和 reduce()函數 構成了函數式程序員工具包的核心。 純高階函數成了函數式方法的主力。事實上,它們是純函數和高階函數應該仿效的典型。它們用一個函數作為輸入,返回沒有副作用的輸出。

這些 JavaScript 函數對每一個函數式程序來說都是至關重要的。他們可以去除循環(huán)和語句,使得代碼更加整潔。這些都是實現 ECMAScript5.1 的瀏覽器的標準,他們只處理數組。每次調用都會創(chuàng)建創(chuàng)建并返回一個新的數組。已存在的數組不會被修改。但是稍等,事情很不止于此。。。他們還將函數作為輸入參數,通常是作為回調的匿名函數。他們會遍歷將整個數組并且將該回調函數應用與每一項!

myArray = [1,2,3,4];

newArray = myArray.map(function(x) {return x*2});

console.log(myArray); // Output: [1,2,3,4]

console.log(newArray); // Output: [2,4,6,8]

除了這三個函數,還有很多函數可以扎入到幾乎每一個函數式應用里:

forEach(),concat(), reverse(), sort(), every() 以及some().

JavaScript的范式

JavaScript當然不是嚴格意義上的函數式編程語言,這也促使了對其他范式的使用:

  • 命令式編程:基于詳細操作描述式的編程

  • 基于原型的面向對象式編程:基于原型對象及其實例的編程

  • 元程序編程:操縱JavsScript執(zhí)行模型的編程方式。對元程序編程的一個很好的定義描述為“編程發(fā)生在您書寫代碼做某事的時候,而元程序編程則發(fā)生在您書寫代碼導致某事的解釋方式發(fā)生變化的時候。

原文地址:https://dzone.com/articles/functional-programming-using-javascript-part-1

翻譯:simei231, 祝青, 李中凱

相關文章

  • mvc 、bootstrap 結合分布式圖簡單實現分頁

    mvc 、bootstrap 結合分布式圖簡單實現分頁

    這篇文章主要介紹了mvc 、bootstrap 結合分布式圖簡單實現分頁的相關資料,非常不錯,具有參考借鑒價值,需要的朋友可以參考下
    2016-10-10
  • js 圖片隨機不定向浮動的實現代碼

    js 圖片隨機不定向浮動的實現代碼

    這篇文章介紹了js圖片隨機不定向浮動的實現代碼,有需要的朋友可以參考一下
    2013-07-07
  • 淺析js中的every()對空數組總返回true

    淺析js中的every()對空數組總返回true

    JavaScript?語言的核心部分足夠大,以至于我們很容易誤解其某些部分的工作方式,本文就來和大家一起討論下為什么JS中的?every()對空數組總返回?true,需要的可以參考下
    2023-09-09
  • 用JS實現HTML標簽替換效果

    用JS實現HTML標簽替換效果

    用JS實現HTML標簽替換效果...
    2007-06-06
  • Javascript中的作用域和上下文深入理解

    Javascript中的作用域和上下文深入理解

    這篇文章主要介紹了Javascript中的作用域和上下文深入理解,本文講解了作用域 VS 上下文、變量作用域、“this”上下文、執(zhí)行上下文(Execution Context)、作用域鏈等內容,需要的朋友可以參考下
    2015-07-07
  • javascript二維數組轉置實例

    javascript二維數組轉置實例

    這篇文章主要介紹了javascript二維數組轉置方法,實例分析了數組行列交換的轉置技巧,具有一定參考借鑒價值,需要的朋友可以參考下
    2015-01-01
  • 如何讓你的JavaScript函數更加優(yōu)雅詳解

    如何讓你的JavaScript函數更加優(yōu)雅詳解

    在Js世界中有些操作會讓你無法理解,但是卻無比優(yōu)雅,下面這篇文章主要給大家介紹了關于如何讓你的JavaScript函數更加優(yōu)雅的相關資料,需要的朋友可以參考下
    2021-07-07
  • JavaScript防抖與節(jié)流超詳細全面講解

    JavaScript防抖與節(jié)流超詳細全面講解

    在開發(fā)中我們經常會遇到一些高頻操作,比如:鼠標移動,滑動窗口,鍵盤輸入等等,節(jié)流和防抖就是對此類事件進行優(yōu)化,降低觸發(fā)的頻率,以達到提高性能的目的。本文就教大家如何實現一個讓面試官拍大腿的防抖節(jié)流函數,需要的可以參考一下
    2022-10-10
  • javascript 偽數組實現方法

    javascript 偽數組實現方法

    能通過Array.prototype.slice轉換為真正的數組的帶有l(wèi)ength屬性的對象。
    2010-10-10
  • 跟我學習javascript的作用域與作用域鏈

    跟我學習javascript的作用域與作用域鏈

    跟我學習javascript的作用域與作用域鏈,感興趣的小伙伴們可以參考一下
    2015-11-11

最新評論