簡介
函數式編程語言在學術領域已經存在相當長一段時間了,但是從歷史上看,它們沒有豐富的工具和庫可供使用。隨著 .NET 平臺上的 Haskell 的出現,函數式編程變得更加流行。一些傳統的編程語言,例如 C++ 和 JavaScript,引入了由函數式編程提供的一些構造和特性。在許多情況下,JavaScript 的重復代碼導致了一些拙劣的編碼。如果使用函數式編程,就可以避免這些問題。此外,可以利用函數式編程風格編寫更加優美的回調。
![]() |
|
因為函數式編程采用了完全不同的組織程序的方式,所以那些習慣于采用命令式范例的程序員可能會發現函數式編程有點難學。在這篇文章中,您將了解一些關于如何采用函數式風格,用 JavaScript 編寫良好的、優美的代碼的示例。我將討論:
![]() ![]() |
![]()
|
函數式編程概念
![]() |
|
在那些通過描述 “如何做” 指定解決問題的方法的語言中,許多開發人員都知道如何進行編碼。例如,要編寫一個計算階乘的函數,我可以編寫一個循環來描述程序,或者使用遞歸來查找所有數字的乘積。在這兩種情況下,計算的過程都在程序中進行了詳細說明。清單 1 顯示了一個計算階乘的可能使用的 C 代碼。
|
這類語言也叫做過程性 編程語言,因為它們定義了解決問題的過程。函數式編程與這個原理有顯著不同。在函數式編程中,需要描述問題 “是什么”。 函數式編程語言又叫做聲明性 語言。同樣的計算階乘的程序可以寫成所有到 n 的數字的乘積。計算階乘的典型函數式程序看起來如 清單 2 中的示例所示。
|
第二個語句指明要得到從 1 開始的前 n 個數字的列表(take n [1..]
),然后找出它們的乘積,1 為基元。這個定義與前面的示例不同,沒有循環或遞歸。它就像階乘函數的算術定義。一旦了解了庫函數(take
和 foldr
)和標記(list notation [ ]
)的意義,編寫代碼就很容易,而且可讀性也很好。
![]() |
|
從歷史上看,函數式編程語言不太流行有各種原因。但是最近,有些函數式編程語言正在進入計算機行業。其中一個例子就是 .NET 平臺上的 Haskell。其他情況下,現有的一些語言借用了函數式編程語言中的一些概念。一些 C++ 實現中的迭代器和 continuation,以及 JavaScript 中提供的一些函數式構造(functional construct),就是這種借用的示例。但是,通過借用函數式構造,總的語言編程范例并沒有發生變化。JavaScript 并沒因為函數式構造的添加就變成了函數式編程語言。
我現在要討論 JavaScript 中的函數式構造的各種美妙之處,以及在日常編碼和工作中使用它們的方式。我們將從一些基本功能開始,然后用它們查看一些更有趣的應用。
匿名函數
在 JavaScript 中,可以編寫匿名函數或沒有名稱的函數。為什么需要這樣的函數?請繼續往下讀,但首先我們將學習如何編寫這樣一個函數。如果擁有以下 JavaScript 函數:
清單 3. 典型的函數
|
然后對應的匿名函數看起來應當如下所示:
清單 4. 匿名函數
|
要使用它,則需要編寫以下代碼:
|
使用函數作為值
也可以將函數作為值使用。還可以擁有一些所賦值是函數的變量。在最后一個示例中,還可以執行以下操作:
清單 6. 使用函數賦值
|
在上面 清單 6 的示例中,為變量 sum 賦的值是函數定義本身。這樣,sum 就成了一個函數,可以在任何地方調用。
調用函數的不同方法
JavaScript 允許用兩種方式調用函數,如清單 7 和 8 所示。
|
或
|
所以也可以編寫以下代碼:
|
可以在括號中編寫函數表達式,然后傳遞給參數,對參數進行運算。雖然在 清單 8 的示例中,有直接包含在括號中的函數名稱,但是按 清單 9 中所示方式使用它時,就不是這樣了。
將函數作為參數傳遞給其他函數
也可以將函數作為參數傳遞給其他函數。雖然這不是什么新概念,但是在后續的示例中大量的使用了這個概念??梢詡鬟f函數參數,如 清單 10 所示。
|
執行最后一個 alert 語句輸出了一個大小為 12 的值。
使用函數式概念
前一節介紹了一些使用函數式風格的編程概念。所給的示例并沒有包含所有的概念,它們在重要性方面也沒有先后順序,只是一些與這個討論有關的概念而已。下面對 JavaScript 中的函數式風格作一快速總結:
這一節將介紹一些有效使用這些概念編寫優美的 JavaScript 代碼的示例。(使用 JavaScript 函數式風格,可以做許多超出這個討論范圍的事。)
清單 11. 比較函數
|
要得到需要的函數,請使用 清單 12 的示例。
|
其中 arr 是類型數組對象。排序函數會根據 arr 數組中對象的日期對所有對象進行排序。比較函數和它的定義一起被傳遞給排序函數,以完成排序操作。使用這個函數:
qsort
函數類似。
清單 13 中的代碼風格通常被用來從數組生成動態 HTML。
|
可以用 清單 14 的代碼替換這個代碼。
|
我使用 Array
類型的 prototype 屬性定義新函數 fold?,F在可以在后面定義的任何數組中使用該函數。
window.setTimeout
函數,該函數有兩個參數。第一個參數是在第二個參數表示的毫秒數之后被調用的函數。清單 15 顯示了完成此操作的一種方法。
|
清單 16 顯示了完成此操作的更好的方式。
|
在處理事件時,如果想在調用完一個回調之后再調用一個回調,也可以使用 清單 16 中的代碼擴展。這可能是一個需要您自行完成的一個練習,現在您的興趣被點燃了吧。
![]() ![]() |
![]()
|
結束語
在許多領域中都可以應用 JavaScript 中的函數式編程,以優美的方式完成日?;顒?。這篇文章中的示例只介紹了幾種情況。如果您找到了函數式編程的合適場景,并應用這些概念,那么您就會有更多的理解,并且可以增加您的優美程度。