1. 程式人生 > >C#函數式編程中的部分應用詳解

C#函數式編程中的部分應用詳解

int float sha idt 這一 文章 thumbnail 好的 sta


何謂函數式編程

相信大家在實際的開發中,很多情況下完成一個功能都需要借助多個類,那麽我們這裏的基本單元就是類。而函數式編程則更加細化,致使我們解決一個功能的基本單元是函數,而不是類,每個功能都是由多個函數構成,並且函數之間沒有直接的關系。如果簡單的文字描述還不足以讓你理解,下面我們就配以圖來演示。

如下圖所示,圖左是我們設計好的三個函數,而右邊則是我們需要實現的功能。而我們需要做的就是利用這三個函數去完成對應的三個功能,筆者在這裏只是進行簡單而又形象的表述,實際的開發過程可能需要更多的函數,並且需要使用不同的函數式編程的方式組合才能完成對應的功能。

技術分享圖片

後面我們假設F1和F2進行組合可以完成功能G1,那麽結果就如下圖所示:

技術分享圖片

對應的其他功能我們依然是按照上面的方式進行組合就可以完成對應的功能,這樣做必然有其對應的優點,對筆者而言最大的優點就是函數不受外部環境的影響,這裏我們不能與類中的方法相提並論,因為方法會受到類上下文變量的影響,特別是在多線程的情況下會出現共享讀和寫的問題,而函數則不會,因為他只是通過參數的方式接收外部的變量,還有一點就是復用性很強,如果前期設計的充分,在後期開發過程中函數可以發揮到最大的作用。說了這麽多廢話,下面我們就可以開始我們的函數式編程的第一部分――部分應用。


部分應用

各位不用被這個名詞嚇壞,他主要是將我們多個參數的函數進行拆分,拆成多個只有一個參數的函數,比如下面這個函數,我們正常寫的話都是這樣寫的:


代碼如下:


Func<int, int, int> Add = (x, y) => x + y;

怎麽調用相信筆者就不需要過多介紹了,下面我們就要讓他能夠支持部分應用:


代碼如下:


Func<int, Func<int, int>> Add = x => y => x + y;

這下就應該明白了吧,只是在接收了一個值之後返回了下一個函數,然後我們再調用這個返回的函數就完成整個調用,我們是不是部分使用了這個函數?所以叫部分應用。下面我們來看看怎麽使用這個函數:


代碼如下:


var Add2 = Add(2);

var result = Add2(4);

這樣分成兩行比較容易看懂,但是我們可以僅僅使用一行就可以了,比如下面這個方式:


代碼如下:


var result = Add(2)(5);

哇,是不是瞬間感覺高大上了,如果我們這個方法的參數再多點,就是括號加括號,相信別人看到你這行代碼後就會呵呵了,然後心裏一萬個“某某”馬奔騰。

我去,看到這的人會可能會吹噓這又沒有什麽太特別的東西,就是函數返回函數。對就是函數返回函數,但是實際運用起來你就會發現舒暢多了,下面筆者簡單的舉一個比較靠譜的例子來說明部分應用能夠帶給我們什麽,比如我們經常需要執行SQL語句,當然需要使用SqlConnection,然後附加上對應的SQL語句,為此我們可以開發一個簡單的函數,用來簡化這一過程:


代碼如下:


Func<SqlConnection, Func<String, DataSet>> ExecSql = x => y =>

{

using (x)

{

x.Open();

var com = x.CreateCommand();

DataSet ds = new DataSet();

com.CommandText = y;

SqlDataAdapter adapter = new SqlDataAdapter(com);

adapter.Fill(ds);

return ds;

}

};

然後調用起來就簡單多了,我們只要傳遞給對應的SqlConnection對象,然後對應的返回值我們就可以用來執行我們的SQL語句了,具體的使用示例如下所示:


代碼如下:


var esql = ExecSql(new SqlConnection(“xxx”));

var rds = esql(“select xxxx from xxx”);

rds = esql(“select ffff from ffff”);

但是做到這還沒有結束,面對那些總是想出奇怪問題的人,我們還有一個需要做,就是我們可能先要傳遞SQL語句,然後再傳遞對應的SqlConnection對象,沒問題,我們專門為此寫個函數:


代碼如下:


Func<String, Func<SqlConnection, DataSet>> ExecSqlT = x => y => ExecSql(y)(x);

我們就繼續該怎麽調用就調用吧,但是上面都是從一開始就利用部分應用的方式來寫,實際情況可能是已經寫好的普通的方式,需要轉換成部分應用的方式。那麽下面我們可以自己先手動的寫幾個擴展,以便於以後的使用,首先我們來寫存在兩個參數和返回值的擴展:


代碼如下:


public static class Functional

{

public static Func<T1, Func<T2, T3>> Currey<T1, T2, T3>(this Func<T1, T2, T3> func)

{

return x => y => func(x, y);

}

}

有了這個擴展之後我們再把上面的例子改寫:


代碼如下:


var ExecSql = Functional.Currey<SqlConnection, String, DataSet>((x, y) =>

{

using (x)

{

x.Open();

var com = x.CreateCommand();

DataSet ds = new DataSet();

com.CommandText = y;

SqlDataAdapter adapter = new SqlDataAdapter(com);

adapter.Fill(ds);

return ds;

}

});

這樣我們就可以按照我們正常的形式來寫,然後調用Functional的Currey就可以了,當然這裏需要顯示的傳遞泛型參數,有些情況下則不需要。

如果需要擴展更多參數的可以對應的寫下去就可以了。當然上面僅僅只是針對沒有參數的情況,我們也可以對Action也進行擴展:


代碼如下:


public static Func<T1, Action<T2>> Currey<T1, T2>(this Action<T1, T2> func)

{

return x => y => func(x, y);

}

到此我們就解決了將普通函數轉換成部分應用方式的函數,但是問題就來了。如果我們一開始寫的是部分應用方式的函數,怎麽將其轉換成普通的函數呢?自然我們還需要下面的擴展能夠將其轉換回去:


代碼如下:


public static Func<T1, T2, T3> UnCurrey<T1, T2, T3>(this Func<T1, Func<T2, T3>> func)

{

return (x, y) => func(x)(y);

}

除聲明外,跑步客文章均為原創,轉載請以鏈接形式標明本文地址
C#函數式編程中的部分應用詳解

本文地址: http://www.paobuke.com/develop/c-develop/pbk23159.html






相關內容

技術分享圖片詳解C#中==、Equals、ReferenceEquals的區別技術分享圖片C#編程總結(六)詳解異步編程技術分享圖片C#檢查foreach判讀是否為null的方法技術分享圖片C#將HashTable中鍵列表或值列表復制到一維數組的方法
技術分享圖片C#預定義數據類型之值類型和引用類型介紹技術分享圖片C#精確計算年齡的方法分析技術分享圖片C#中委托和事件的區別實例解析技術分享圖片WinForm中DefWndProc、WndProc與IMessageFilter的區別

C#函數式編程中的部分應用詳解