国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > C# > 正文

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

2020-01-24 02:09:56
字體:
來源:轉載
供稿:網友

何謂函數式編程

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

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

后面我們假設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);
}

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 中方县| 斗六市| 报价| 泾阳县| 凯里市| 墨江| 凤翔县| 德钦县| 嘉兴市| 五华县| 郑州市| 星子县| 庆城县| 安多县| 淄博市| 洞口县| 新宁县| 西盟| 泰州市| 西乌| 屏南县| 江永县| 砚山县| 同德县| 巴彦淖尔市| 昌邑市| 四川省| 车致| 锡林浩特市| 新余市| 东阳市| 肇源县| 淮滨县| 汪清县| 朝阳区| 灌南县| 合水县| 奇台县| 阳泉市| 隆回县| 河南省|