自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

C#2.0之殤,以及函數(shù)式編程的未來

開發(fā) 后端
本文通過對(duì)C#2.0和C#3.0的比較,探討了函數(shù)式語言的優(yōu)勢(shì)。有人說函數(shù)式編程有什么用,C#3.0就是個(gè)很好的證明。

似乎還有不少項(xiàng)目在用C#2.0,但是C#2.0的生產(chǎn)力實(shí)在不如C#3.0——如果您不信,那么一會(huì)兒就會(huì)意識(shí)到這一點(diǎn)。有朋友認(rèn)為語言能力不重要,有了好用的框架/類庫也可以有很高的生產(chǎn)力。所以這篇文章,我們就設(shè)法使用“類庫”來彌補(bǔ)C#2.0的缺陷。

但是,我們真做的到嗎?

C#2.0之殤

C#2.0較C#1.0來說是一個(gè)突破,其中引入了泛型,以及匿名方法等新特性。如果前者還可以說是平臺(tái)的增強(qiáng),而語言只是個(gè)“輔助”的話,而后者則百分之一百是編譯器的魔法了。別小看這個(gè)特性,它為C#3.0的高生產(chǎn)力踏出了堅(jiān)實(shí)的一步——不過還是差了很多。例如,我們有一個(gè)要求:“把一個(gè)字符串?dāng)?shù)組中的元素轉(zhuǎn)化為整數(shù),再將其中的偶數(shù)放入一個(gè)List< int>容器中”。如果是C#3.0,這是再簡單不過的功能:

  1. string[]strArray={"1","2","3","4"};  
  2. vareven=strArray.Select(s=>Int32.Parse(s)).Where(i=>i%2==0).ToList();  

那么對(duì)于C#2.0(當(dāng)然對(duì)于C#1.0也一樣),代碼又該怎么寫呢?

  1. List< int>even=newList< int>();  
  2. foreach(stringsinstrArray)  
  3. {  
  4. inti=Int32.Parse(s);  
  5. if(i%2==0)  
  6. {  
  7. even.Add(i);  
  8. }  
  9. }  

有人說函數(shù)式編程有什么用,C#3.0就是個(gè)很好的證明。C#3.0中引入了Lambda表達(dá)式,增強(qiáng)了在語言中構(gòu)造匿名方法的能力——這是一個(gè)語言中函數(shù)式編程特性的必備條件。C#3.0的實(shí)現(xiàn)與C#2.0相比,可讀性高,可以直接看出轉(zhuǎn)化、過濾,以及構(gòu)造容器的過程和標(biāo)準(zhǔn)。由于語言能力的增強(qiáng),程序的表現(xiàn)能力得到了很大的提高,在很多時(shí)候,我們可以省去將一些代碼提取為獨(dú)立方法的必要。當(dāng)然,即使您將其提取為額外的方法,C#3.0也可以讓您寫出更少的代碼。

如果您覺得以上代碼的差距還不是過于明顯的話——那么以下功能呢?

  1. int[]intArray={1,2,3,4,5,6,7,8,9,10};  
  2.  
  3. //所有偶數(shù)的平均數(shù)  
  4.  
  5. varevenAverage=intArray.Where(i=>i%2==0).Average();  
  6.  
  7. //都是偶數(shù)?  
  8.  
  9. varallEven=intArray.All(i=>i%2==0);  
  10.  
  11. //包含偶數(shù)?  
  12.  
  13. varcontainsEven=intArray.Any(i=>i%2==0);  
  14.  
  15. //第4到第8個(gè)數(shù)  
  16.  
  17. varfourthToEighth=intArray.Skip(3).Take(5);  

如果您使用C#2.0來寫,您會(huì)怎么做?

拯救C#2.0

C#3.0通過引入了函數(shù)式編程特性大幅增強(qiáng)了語言的生產(chǎn)力。如果說C#2.0和Java還沒有太大差距的話,那么C#3.0已經(jīng)將Java甩開地太遠(yuǎn)太遠(yuǎn)。不過真要說起來,在Java中并非不可以加入函數(shù)式編程的理念。只不過,如果沒有足夠的語言特性進(jìn)行支持(如快速構(gòu)造匿名函數(shù)、閉包、一定程度的類型推演等等),函數(shù)式編程對(duì)于某些語言來說幾乎只能成為“理念”。不過現(xiàn)在,我們暫且先放下對(duì)“函數(shù)式編程”相關(guān)內(nèi)容的探索,設(shè)法拯救C#2.0所缺失的生產(chǎn)力吧。

C#3.0中可以使用Lambda表達(dá)式構(gòu)造一個(gè)匿名函數(shù),這個(gè)能力其實(shí)在C#2.0中也有。我們姑且認(rèn)為這點(diǎn)不是造成差距的主要原因,那么有一點(diǎn)是C#2.0絕對(duì)無法實(shí)現(xiàn)的,那就是“擴(kuò)展方法”。C#3.0中的擴(kuò)展方法,可以“零耦合”地為一個(gè),甚至一系列類型添加“實(shí)例方法”。當(dāng)然,這也是編譯器的功能,實(shí)際上我們只是定義了一些靜態(tài)方法而已。這一點(diǎn)在C#2.0中還是可以做到的:

  1. publicclassEnumerable  
  2.  
  3. {  
  4.  
  5. publicstaticIEnumerable< T>Where< T>(Func< T,bool>predicate,IEnumerable< T>source)  
  6.  
  7. {  
  8.  
  9. foreach(Titeminsource)  
  10.  
  11. {  
  12.  
  13. if(predicate(item))  
  14.  
  15. {  
  16.  
  17. yieldreturnitem;  
  18.  
  19. }  
  20.  
  21. }  
  22.  
  23. }  
  24.  
  25. publicstaticIEnumerable< TResult>Select< T,TResult>(Func< T,TResult>selector,IEnumerable< T>source)  
  26.  
  27. {  
  28.  
  29. foreach(Titeminsource)  
  30.  
  31. {  
  32.  
  33. yieldreturnselector(item);  
  34.  
  35. }  
  36.  
  37. }  
  38.  
  39. publicstaticList< T>ToList< T>(IEnumerable< T>source)  
  40.  
  41. {  
  42.  
  43. List< T>list=newList< T>();  
  44.  
  45. foreach(Titeminsource)  
  46.  
  47. {  
  48.  
  49. list.Add(item);  
  50.  
  51. }  
  52.  
  53. returnlist;  
  54.  
  55. }  
  56.  
  57. }  

于是現(xiàn)在,我們便可以換種寫法來實(shí)現(xiàn)相同的功能了:

  1. string[]strArray={"1","2","3","4"};  
  2.  
  3. List< int>even=  
  4.  
  5. Enumerable.ToList(  
  6.  
  7. Enumerable.Where(  
  8.  
  9. delegate(inti){returni%2==0;},  
  10.  
  11. Enumerable.Select(  
  12.  
  13. delegate(strings){returnInt32.Parse(s);},  
  14.  
  15. strArray)));  

即使您可以接受delegate關(guān)鍵字構(gòu)造匿名函數(shù)的能力,但是上面的做法還是有個(gè)天生的缺陷:邏輯與表現(xiàn)的次序想反。我們想表現(xiàn)的邏輯順序?yàn)椋恨D(zhuǎn)化(Select)、過濾(Where)、及容器構(gòu)造(ToList),C#3.0所表現(xiàn)出的順序和它相同,而C#2.0的順序則相反。由于語言能力的缺失,這個(gè)差距無法彌補(bǔ)。很多時(shí)候,語言的一些“小功能”并不能說是可有可無的特性,它很可能直接決定了是否可以用某種語言來構(gòu)造InternalDSL或進(jìn)行BDD。例如,由于F#的靈活語法,F(xiàn)sTest使得開發(fā)人員可以寫出"foobar"|>shouldcontains"foo"這樣的語句來避免機(jī)械的Assert語法。同樣,老趙也曾經(jīng)使用actor< =msg這樣的邏輯來替代actor.Post(msg)的顯式調(diào)用方式。

封裝邏輯

既然沒有“擴(kuò)展方法”,我們要避免靜態(tài)方法的調(diào)用形式,那么就只能在一個(gè)類中定義邏輯了。這點(diǎn)并不困難,畢竟在API的設(shè)計(jì)發(fā)展至今,已經(jīng)進(jìn)入了關(guān)注FluentInterface的階段,這方面已經(jīng)積累了大量的實(shí)踐。于是我們構(gòu)造一個(gè)Enumerable< T>類,封裝IEnumerable< T>對(duì)象,以此作為擴(kuò)展的入口:

  1. publicclassEnumerable< T>  
  2.  
  3. {  
  4.  
  5. privateIEnumerable< T>m_source;  
  6.  
  7. publicEnumerable(IEnumerable< T>source)  
  8.  
  9. {  
  10.  
  11. if(source==null)thrownewArgumentNullException("source");  
  12.  
  13. this.m_source=source;  
  14.  
  15. }  
  16.  
  17. ...  
  18.  
  19. }  
  20.  
  21. 并以此定義所需的Select和Where方法:  
  22.  
  23. publicEnumerable< T>Where(Func< T,bool>predicate)  
  24.  
  25. {  
  26.  
  27. if(predicate==null)thrownewArgumentNullException("predicate");  
  28.  
  29. returnnewEnumerable< T>(Where(this.m_source,predicate));  
  30.  
  31. }  
  32.  
  33. privatestaticIEnumerable< T>Where(IEnumerable< T>source,Func< T,bool>predicate)  
  34.  
  35. {  
  36.  
  37. foreach(Titeminsource)  
  38.  
  39. {  
  40.  
  41. if(predicate(item))  
  42.  
  43. {  
  44.  
  45. yieldreturnitem;  
  46.  
  47. }  
  48.  
  49. }  
  50.  
  51. }  
  52.  
  53. publicEnumerable< TResult>Select< TResult>(Func< T,TResult>selector)  
  54.  
  55. {  
  56.  
  57. if(selector==null)thrownewArgumentNullException("selector");  
  58.  
  59. returnnewEnumerable< TResult>(Select(this.m_source,selector));  
  60.  
  61. }  
  62.  
  63. privatestaticIEnumerable< TResult>Select< TResult>(IEnumerable< T>source,Func< T,TResult>selector)  
  64.  
  65. {  
  66.  
  67. foreach(Titeminsource)  
  68.  
  69. {  
  70.  
  71. yieldreturnselector(item);  
  72.  
  73. }  
  74.  
  75. }  

這些擴(kuò)展都是些高階函數(shù),也都有延遲效果,相信很容易理解,在此就不多作解釋了。在這里我們直接觀察其使用方式:

  1. List< int>even=newEnumerable< string>(strArray)  
  2.  
  3. .Select(delegate(strings){returnInt32.Parse(s);})  
  4.  
  5. .Where(delegate(inti){returni%2==0;})  
  6.  
  7. .ToList();  

不知道您對(duì)此有何感覺?

老趙對(duì)此并不滿意,尤其是和C#3.0相較之下。我們雖然定義了Enumerable封裝類,并提供了Select和Where等邏輯,但是由于匿名函數(shù)的構(gòu)造還是較為丑陋。使用delegate構(gòu)造匿名函數(shù)還是引起了不少噪音:

與JavaScript的function關(guān)鍵字,和VB.NET的Function關(guān)鍵字一樣,C#2.0在構(gòu)造匿名函數(shù)時(shí)無法省確delegate關(guān)鍵字。

與C#3.0中的Lambda表達(dá)式相比,使用delegate匿名函數(shù)缺少了必要的類型推演。

使用delegate構(gòu)造匿名函數(shù)時(shí)必須提供完整的方法體,也就是只能提供“語句”,而不能僅為一個(gè)“表達(dá)式”,因此return和最后的分號(hào)無法省確。

我們?cè)O(shè)法拯救C#2.0,但是我們真的做到了嗎?

框架/類庫真能彌補(bǔ)語言的生產(chǎn)力嗎?

【編輯推薦】

  1. 淺談CLR 4.0安全模型的運(yùn)作機(jī)制
  2. 探秘CLR 4.0中的代碼契約
  3. CLR線程池的作用與原理淺析
  4. SQL Server 2005中的CLR集成
  5. CLR 4.0中的新內(nèi)容 狀態(tài)錯(cuò)亂異常
責(zé)任編輯:yangsai 來源: 博客園
相關(guān)推薦

2010-02-24 11:20:23

C#

2020-11-01 09:05:16

函數(shù)式編程編程數(shù)據(jù)分析

2015-07-15 15:24:25

OpenFlowSDN

2011-05-20 10:12:14

接入寬帶互聯(lián)網(wǎng)

2013-05-31 02:16:07

創(chuàng)業(yè)程序員創(chuàng)業(yè)

2018-10-08 13:41:01

運(yùn)營商通信網(wǎng)絡(luò)攜號(hào)轉(zhuǎn)網(wǎng)

2013-09-09 09:41:34

2012-09-21 09:21:44

函數(shù)式編程函數(shù)式語言編程

2021-04-06 11:04:54

網(wǎng)絡(luò)安全C語言代碼

2011-03-08 15:47:32

函數(shù)式編程

2016-10-31 20:46:22

函數(shù)式編程Javascript

2020-09-24 10:57:12

編程函數(shù)式前端

2025-03-11 10:00:20

Golang編程函數(shù)

2024-02-28 08:37:28

Lambda表達(dá)式Java函數(shù)式接口

2023-12-14 15:31:43

函數(shù)式編程python編程

2011-08-24 09:13:40

編程

2022-09-22 08:19:26

WebFlux函數(shù)式編程

2013-03-05 10:01:29

Python函數(shù)式編程

2013-03-04 10:03:17

Python函數(shù)式編程

2013-03-04 09:47:08

Python函數(shù)式編程
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)