這篇文章主要介紹了javascript函數式編程,實例分析了javascript函數式編程的相關使用技巧,非常具有實用價值,需要的朋友可以參考下
本文實例講述了javascript函數式編程。分享給大家供大家參考。具體分析如下:
js像其他動態語言一樣是可以寫高階函數的,所謂高階函數是可以操作函數的函數。因為在js中函數是一個徹徹底底的對象,屬于第一類公民,這提供了函數式編程的先決條件。
下面給出一個例子代碼,出自一本js教程,功能是計算數組元素的平均值和標準差,先列出非函數式編程的一種寫法:
- var data = [1,1,3,5,5];
- var total = 0;
- for(var i = 0;i < data.length;i++)
- total += data[i];
- var mean = tatal/data.length; //平均數為3
- //計算標準差
- total = 0;
- for(var i = 0;i < data.length;i++){
- var deviation = data[i] - mean;
- tatal += deviation * deviation;
- }
- var stddev = Math,.sqrt(total/(data.length-1));//標準差為2
為了使用函數式編程,我們預先定義一些幫助函數(helper functions):
- //將類數組對象轉換為真正的數組
- function array(a,n){
- return Array.prototype.slice.call(a,n||0);
- }
- //將函數實參傳遞至左側
- function partial_left(f){
- var args = arguments;
- return function(){
- var a = array(args,1);
- a = a.concat(array(arguments));
- return f.apply(this,a);
- };
- }
- //將函數的實參傳遞至右側
- function partial_right(f){
- var args = arguments;
- return function(){
- var a = array(arguments);
- a = a.concat(array(args,1));
- return f.apply(this,a);
- };
- }
- //該函數實參被用做模版,
- //實參列表中的undefined值會被實際實參值填充。
- function partial(f){
- var args = arguments;
- return function(){
- var a = array(args,1);
- var i = 0,j = 0;
- for(;i<a.length;i++)
- if(a[i] === undefined)
- a[i] = arguments[j++];
- a = a.concat(array(arguments,j));
- return f.apply(this,a);
- };
- }
- //返回一個函數類似于f(g())
- function compose(f,g){
- return function(){
- return f.call(this,g.apply(this,arguments));
- };
- }
下面我們給出完全用函數式編程的js代碼:
- var data = [1,1,3,5,5];
- var sum = function(x,y){return x+y;};
- var product = function(x,y){return x*y;};
- var neg = partial(product,-1);
- var square = partial(Math.pow,undefined,2);
- var sqrt = partial(Math.pow,undefined,0.5);
- var reciprocal = partial(Math.pow,undefined,-1);
- //好吧,高潮來鳥 :)
- var mean = product(reduce(data,sum),reciprocal(data.length));
- var stddev = sqrt(product(reduce(map(data,compose(square,partial(sum,neg(mean)))),sum),reciprocal(sum(data.length,-1))));
除了reduce和map函數,其他函數前面都給出了。reduce函數類似與ruby中的inject函數:
- ary = (1..10).to_a
- ary.inject(0) {|sum,i|sum + i} //結果為55
js的寫法如下:
- var ary = [1,2,3,4,5,6,7,8,9,10]
- ary.reduce(function(sum,i){
- return sum + i;
- },0);
0為sum的初始值,如果省略則sum為數組第一個元素的值,這里可以省略。
map函數也很簡單,類似與對數組的每一個元素做操作,然后返回一個經過操作后的數組,就以ruby代碼為例,js代碼與此類似:
- a = (1..3).to_a; #數組[1,2,3]
- a.map {|x| x*2} #返回新數組[2,4,6]
下面我們來分析下那一長串的代碼:)
sum和product定義了元素相加和相乘的函數;
neg也是一個函數功能等價于:product(-1,x),即對x值求負;
square函數等價于:Math.pow(x,2),即計算x的平方值,注意這里partial的第二個參數是undefined,這意味著這里的形參會被第一個實參填補;再說的明白點:square(x)功能等于Math.pow(x,2)。
sqrt函數和square類似,功能等價于:Math.pow(x,0.5),相當于計算x的開二次方。
最后一個函數reciprocal也沒什么難度,等價于:Math.pow(x,-1),即計算x的負一次方,相當于計算x的倒數。
下面就是如何把上面各種函數揉捏在一起鳥 :)
先看平均值的計算,很簡單:就是先計算數組元素的和然后乘上數組長度的倒數,即數組和/數組長度。
最后來看貌似很難的標準差,我們最好由內向外看:
先看包含neg的那層:
- //等價于函數sum(-1 * mean + x)
- partial(sum,neg(mean)
下面看compose函數:
- //下面在源代碼上做了等價替換,可以再次等價于:
- //square(sum(-1*mean + x)),再次展開(我剝,我剝,我剝洋蔥...):
- //Math.pow(sum(-1*mean + x),2);
- compose(square,sum(-1*mean + x))
接下來看map函數:
//很清楚吧!?即data中每一個元素都為一個x,將其傳入后面的函數,然后返回一個計算后的新數組,即新數組中的每個元素的值是data中的每個元素加上data負的平均數,然后對其結果計算2次方的結果。
- map(data,Math.pow(sum(-1*mean + x),2))
再接著看map外面的reduce函數:
- //將前面新數組的每個元素值加起來。
- reduce(map(...),sum)
然后看一下reciprocal函數:
- //等價于求(data.length-1)的倒數
- reciprocal(sum(data.length,-1))
再看外層的product函數:
- //等價于新數組元素的和除以(data.length-1)
- product(reduce(...),reciprocal(...))
最外層的sqrt表示對以上除法得出的結果求平方根;大家可以對照一下前面非函數編程的代碼,是一樣一樣滴 :) 看似蠻怕人的一大坨代碼,展開分析后難度立馬將至零。如果各位看官最后表示還是未看明白,那完全是本貓語言表達能力的問題,歡迎提問。
解釋完畢,打完收功,大功告成。
希望本文所述對大家的javascript程序設計有所幫助。
新聞熱點
疑難解答
圖片精選