.net 系列:Expression表達式樹、lambda、匿名委托 的使用

来源:http://www.cnblogs.com/nicholashjh/archive/2017/11/30/7928205.html
-Advertisement-
Play Games

首先定義一個泛型委托類型,如下: 實現泛型委托的主體代碼,並調用: 以上代碼展示了委托類型Function<T>主體定義的四種方式,分別是實名委托、匿名委托、Lambda表達式、expression表達式樹。 從Function<T>委托主體的代碼定義來看是越來越簡單和友好,這些變化很大部分應歸功於 ...


首先定義一個泛型委托類型,如下:

public delegate T Function<T>(T a, T b);

實現泛型委托的主體代碼,並調用:

 

 

 1 public static string Add(string a, string b)
 2           {
 3                 return string.Format("{0} #### {1}",a,b);
 4           }
 5  //實名委托方式
 6  Function<string> func = new Function<string>(Add);
 7  Console.WriteLine( func("hello", "world") );
 8 
 9  //匿名委托方式
10  Function<string> func1 = new Function<string>(delegate(string a, string b) {
11         return string.Format("{0} @@@@ {1}",a,b);
12  });
13  Console.WriteLine(func1("hello", "world"));
14 
15  //Lambda表達式方式
16  Function<string> func2 = (a, b) => string.Format("{0} **** {1}", a, b);
17  Console.WriteLine(func2("hello", "world"));
18 
19  Expression<Function<string>> func2_0;
20  //func2_0 = func;  //不支持將委托直接賦值給表達式樹
21  //func2_0 = func1; //不支持將委托直接賦值給表達式樹
22  //func2_0 = func2; //不支持將委托直接賦值給表達式樹
23 
24  //(a, b) => string.Format("{0} **** {1}", a, b)語句塊的類型是lambda expression,即我們常說的lambda表達式
25  //所以,func2_0 = (a, b) => string.Format("{0} **** {1}", a, b)的直接賦值是沒有問題的。
26  func2_0 = (a, b) => string.Format("{0} **** {1}", a, b);
27  Console.WriteLine(func2_0.Compile()("hello", "world"));

 

以上代碼展示了委托類型Function<T>主體定義的四種方式,分別是實名委托、匿名委托、Lambda表達式、expression表達式樹。

從Function<T>委托主體的代碼定義來看是越來越簡單和友好,這些變化很大部分應歸功於C#的語法糖。

總結:不管委托主體在編寫的形式上怎麼簡化,但依然改變不了它委托類型的本質,當委托代碼塊被調用時會即時執行。

隨著C#的發展,後來加入了expression這個東東,簡稱表達式樹,我想用過ling to sql、linq to entity、linq to xml等等的你是不會陌生的。
expression是一種數據結構,我們可以將平常編寫的C#語句塊(或者叫表達式)的各部分進行分解並存入這個樹結構當中,保存在expression樹結構中的語句塊是不能直接執行的。
當我們需要將expression結構中的數據抽取並還原時就需要調用expression.Compile()方法,這裡我稱之為編譯。編譯後得到的結果就是我們之前存入的語句塊,這是數據結構還原成語句塊的過程(這是一個比喻)。
當然將數據還原成語句塊時依據解析引擎的不同會產生不同的輸出結果,如果引擎是linq to sql那麼解析後輸出的就是可供資料庫執行的sql,如果引擎是linq to xml則解析後輸出的是Xpath之類的表達式(沒親自驗證)

下麵就請你和我一起來體驗一下expression表達式數據的存儲和編譯輸出吧!!!!仍以上面的場景為例子

 1 //expression表達式樹主體構造開始
 2 ParameterExpression paramA = Expression.Parameter(typeof(object), "a"); //聲明Lambda表達式中的參數表達式a
 3 ParameterExpression paramB = Expression.Parameter(typeof(object), "b"); //聲明Lambda表達式中的參數表達式b
 4 ConstantExpression constantExp = Expression.Constant("{0} !!!!! {1}",typeof(string));//聲明文本塊常量表達式
 5 MethodCallExpression bodyExp = Expression.Call(typeof(string).GetMethod("Format", new Type[] { typeof(string), typeof(object), typeof(object) })
 6                 , new Expression[] { constantExp, paramA, paramB }); //聲明String.Format()方法調用表達式
 7 //expression表達式樹主體構造結束
 8 
 9 
10 //1.構造類型為LambdaExpression的lambda表達式樹,編譯後得到委托的基元類型(弱類型)。
11 LambdaExpression func3 = Expression.Lambda(bodyExp, paramA, paramB);//將以上各個表達式部分組合為Lambda表達式
12 Delegate dg = func3.Compile();//編譯表達式樹得到委托
13 Console.WriteLine(dg.DynamicInvoke("hello", "world"));//調用委托並將結果輸出到控制台
14 //Console.WriteLine(func3.Compile().DynamicInvoke("hello", "world")); //上面兩步可以簡化為這句代碼
15 
16 //2.構造類型為Expression<Function<string>>的泛型lambda表達式樹,編譯後得到委托可直接調用。
17 Expression<Function<string>> func4 = Expression.Lambda<Function<string>>(bodyExp, paramA, paramB);
18 Console.WriteLine(func4.Compile()("xxxx", "yyyy"));
19 
20  //3.構造類型為Expression<Func<string, string, string>>的泛型lambda表達式樹,編譯後得到委托可直接調用。
21  //與上面的區別是這裡用系統定義的Func<in T1, in T2, out TResult>泛型委托代替了自定義的Function<T>委托。
22  Expression<Func<string, string, string>> func5 = Expression.Lambda<Func<string, string, string>>(bodyExp, paramA, paramB);
23  Console.WriteLine(func5.Compile()("yyyy", "zzzz"));
24 
25  //以上總結了expression表達式的創建和調用的不同方式,以下是幾個有關expression的擴展例子
26  //4.動態構造string.Concat("hello", "world")語句塊
27  var concatMethod = typeof(string).GetMethod("Concat", new[] { typeof(string), typeof(string) });
28  var addExpr = Expression.Add(Expression.Constant("hello "), Expression.Constant("world"), concatMethod);
29  Expression<Func<string>> e = Expression.Lambda<Func<string>>(addExpr);
30  Console.WriteLine(e.Compile()());
31 
32  //5.動態構造Math.Sin(100)語句塊
33  ParameterExpression expA = Expression.Parameter(typeof(double), "a"); //參數a
34  MethodCallExpression expCall = Expression.Call(
35         typeof(Math).GetMethod("Sin",new Type[]{typeof(double)}),expA); 
36  LambdaExpression exp = Expression.Lambda(expCall, expA); // a => Math.Sin(a)
37  Console.WriteLine( exp.Compile().DynamicInvoke(100) );
38 
39  //6.動態構造Console.WriteLine("aaa")語句塊
40  ConstantExpression _constExp = Expression.Constant("aaa", typeof(string));//一個常量
41  MethodCallExpression _methodCallexp = Expression.Call(typeof(Console).GetMethod("WriteLine", new Type[] { typeof(string) }), _constExp);
42  Expression<Action> consoleLambdaExp = Expression.Lambda<Action>(_methodCallexp);
43  consoleLambdaExp.Compile()();

 

 

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 整數回覆 整數回覆就是一個以 ":" 開頭, CRLF 結尾的字元串表示的整數。 比如說, ":0\r\n" 和 ":1000\r\n" 都是整數回覆。 返回整數回覆的其中兩個命令是 INCR 和 LASTSAVE 。 被返回的整數沒有什麼特殊的含義, INCR 返回鍵的一個自增後的整數值, 而 L ...
  • 直接計算字元串算式方法: ...
  • 射線:從一個點往一個方向,發射一根無限長的射線,這根射線會與場景中帶有 Collider 組件的物體發生碰撞。 射線的使用: 根據上面的代碼: hitInfo.point:表示碰撞點的坐標。 Physics.Raycast():使用這個方法檢測射線時,因為該方法重載很多,一定要弄清楚自己使用的是哪個 ...
  • 實現代碼(C#) 1、發送GET指令 2、接收批量回覆 3、 結果: 代碼重構 1、發送指令 2、接收回覆 3、GET和SET指令 4、重構後的代碼 是不是簡潔很多??? 5、結果 ...
  • 結論: 1、EF 查詢 比ADO慢,甚至直接報告超時錯誤,原因不詳。 2、在原生ADO.Net中 使用 參數化查詢 比 直接使用sql拼接 慢幾十倍!!! ADO.Net代碼測試 EF代碼(已捨棄) 資料庫內部測試 sqlHelper類 博客園非常蛋疼的說:字數少有150字不能發佈的首頁。 答:親, ...
  • using Newtonsoft.Json; using System; using System.Collections.Concurrent; using System.Diagnostics; using System.IO; using System.Threading; using Sys... ...
  • 文檔中包含圖片的話,會使得整個文檔比較大,占用存儲空間且不利於快速、高效的傳輸文件。針對一些包含大量高質圖片的PDF文檔,若是對圖片進行壓縮,可以有效減少文檔的占用空間。並且,在文檔傳輸過程中也可以減少傳送時間,提高效率。本文將介紹2種通過C#來實現PDF圖片文檔壓縮的方法。使用此方法,需要用到最新 ...
  • 相關下載:https://system.data.sqlite.org/index.html/doc/trunk/www/downloads.wiki 在Visual Studio中要支持訪問SQLite文件數據源,首先需要安裝SQLite .NET的相關組件,安裝完SQLite組件後,就可以在Vi ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...