【轉】由淺入深表達式樹(一)創建表達式

来源:https://www.cnblogs.com/RYouHoo-923/archive/2018/01/12/8275602.html
-Advertisement-
Play Games

為什麼要學習表達式樹?表達式樹是將我們原來可以直接由代碼編寫的邏輯以表達式的方式存儲在樹狀的結構里,從而可以在運行時去解析這個樹,然後執行,實現動態的編輯和執行代碼。LINQ to SQL就是通過把表達式樹翻譯成SQL來實現的,所以瞭解表達樹有助於我們更好的理解 LINQ to SQL,同時如果你有 ...


       為什麼要學習表達式樹?表達式樹是將我們原來可以直接由代碼編寫的邏輯以表達式的方式存儲在樹狀的結構里,從而可以在運行時去解析這個樹,然後執行,實現動態的編輯和執行代碼。LINQ to SQL就是通過把表達式樹翻譯成SQL來實現的,所以瞭解表達樹有助於我們更好的理解 LINQ to SQL,同時如果你有興趣,可以用它創造出很多有意思的東西來。

  表達式樹是隨著.NET 3.5推出的,所以現在也不算什麼新技術了。但是不知道多少人是對它理解的很透徹, 在上一篇Lambda表達式的回覆中就看的出大家對Lambda表達式和表達式樹還是比較感興趣的,那我們就來好好的看一看這個造就了LINQ to SQL以及讓LINQ to Everything的好東西吧。

  本系列計劃三篇,第一篇主要介紹表達式樹的創建方式。第二篇主要介紹表達式樹的遍歷問題。第三篇,將利用表達式樹打造一個自己的LinqProvider。

  本文主要內容:

創建一個簡單的Lambda表達式樹

  在 上一篇Lambda表達式中我們提到了可以直接根據Lambda表達式來創建表達式樹,這應該是最直接的創建表達式樹的方式了。

Expression<Func<int, int>> expr = x => x + 1;
Console.WriteLine(expr.ToString());  // x=> (x + 1)

// 下麵的代碼編譯不通過
Expression<Func<int, int, int>> expr2 = (x, y) => { return x + y; };
Expression<Action<int>> expr3 = x => {  };

  但是別想象的太美好,這種方式只能創建最簡單的表達式樹,複雜點的編譯器就不認識了。

  右邊是一個Lambda表達式,而左邊是一個表達式樹。為什麼可以直接賦值呢?這個就要多虧我們的Expression<TDelegate>泛型類了。而Expression<TDelegate>是直接繼承自LambdaExpression的,我們來看一下Expression的構造函數:

internal Expression(Expression body, string name, bool tailCall, ReadOnlyCollection<ParameterExpression> parameters)
    : base(typeof(TDelegate), name, body, tailCall, parameters)
{
}

  實際上這個構造函數什麼也沒有做,只是把相關的參數傳給了父類,也就是LambdaExpression,由它把我們表達式的主體,名稱,以及參數保存著。

Expression<Func<int, int>> expr = x => x + 1;
Console.WriteLine(expr.ToString());  // x=> (x + 1)

var lambdaExpr = expr as LambdaExpression;
Console.WriteLine(lambdaExpr.Body);   // (x + 1)
Console.WriteLine(lambdaExpr.ReturnType.ToString());  // System.Int32

foreach (var parameter in lambdaExpr.Parameters)
{
    Console.WriteLine("Name:{0}, Type:{1}, ",parameter.Name,parameter.Type.ToString());
}

//Name:x, Type:System.Int32

  

  簡單的來說,Expression<TDelegate>泛型類做了一層封裝,方便我們根據Lambda表達式來創建Lambda表達式樹。它們中間有一個轉換過程,而這個轉換的過程就發生在我們編譯的時候。還記得我們Lambda表達式中講的麽?Lambda表達式在編譯之後是普通的方法,而Lambda式樹是以一種樹的結構被載入到我們的運行時的,只有這樣我們才可以在運行時去遍歷這個樹。但是為什麼我們不能根據Expression<TDelegate>來創建比較複雜的表達式樹呢?您請接著往下看。

創建一個複雜的Lambda表達式樹

  下麵我們就來一步一步的創建一個複雜的表達式樹,你們準備好了麽?上面我們講到直接由Lambda表達式的方式來創建表達式樹,可惜只限於一種類型。下麵我們就來演示一下如何創建一個無參無返回值的表達式樹。

// 下麵的方法編譯不能過 
/*
Expression<Action> lambdaExpression2 = () =>
{
    for (int i = 1; i <= 10; i++)
    {
        Console.WriteLine("Hello");
    }
};
*/     
       
// 創建 loop表達式體來包含我們想要執行的代碼
LoopExpression loop = Expression.Loop(
    Expression.Call(
        null,
        typeof(Console).GetMethod("WriteLine", new Type[] { typeof(string) }),
        Expression.Constant("Hello"))
        );

// 創建一個代碼塊表達式包含我們上面創建的loop表達式
BlockExpression block = Expression.Block(loop);

// 將我們上面的代碼塊表達式
Expression<Action> lambdaExpression =  Expression.Lambda<Action>(block);
lambdaExpression.Compile().Invoke();

  上面我們通過手動編碼的方式創建了一個無參的Action,執行了一組迴圈。代碼很簡單,重要的是我們要熟悉這些各種類型的表達式以及他們的使用方式。上面我們引入了以下類型的表達式:

  看起來神密的表達式樹也不過如此嘛?如果大家去執行上面的代碼,就會陷入死迴圈,我沒有為loop加入break的條件。為了方便大家理解,我是真的一步一步來啊,現在我們就來終止這個迴圈。就像上面那一段不能編譯通過的代碼實現的功能一樣,我們要輸出10個”Hello”。

  上面我們先寫了一個LoopExpression,然後把它傳給了BlockExpresson,從而形成的的一塊代碼或者我們也可以說一個方法體。但是如果我們有多個執行塊,而且這多個執行塊裡面需要處理同一個參數,我們就得在block裡面聲明這些參數了。

ParameterExpression number=Expression.Parameter(typeof(int),"number");
            
BlockExpression myBlock = Expression.Block(
    new[] { number },
    Expression.Assign(number, Expression.Constant(2)),
    Expression.AddAssign(number, Expression.Constant(6)),
    Expression.DivideAssign(number, Expression.Constant(2)));

Expression<Func<int>> myAction = Expression.Lambda<Func<int>>(myBlock);
Console.WriteLine(myAction.Compile()());
// 4

  我們聲明瞭一個int的變數並賦值為2,然後加上6最後除以2。如果我們要用變數,就必須在block的你外面聲明它,並且在block裡面把它引入進來。否則在該表達式樹時會出現,變數不在作用域里的錯。

  下麵我們繼續我們未完成的工作,為迴圈加入退出條件。為了讓大家快速的理解loop的退出機制,我們先來看一段偽代碼:

LabelTarget labelBreak = Expression.Label();
Expression.Loop(
    "如果 條件 成功"
        "執行成功的代碼"
    "否則"
        Expression.Break(labelBreak) //跳出迴圈
    , labelBreak); 

  我們需要藉助於LabelTarget 以及Expression.Break來達到退出迴圈的目地。下麵我們來看一下真實的代碼:

LabelTarget labelBreak = Expression.Label();
ParameterExpression loopIndex = Expression.Parameter(typeof(int), "index");

BlockExpression block = Expression.Block(
new[] { loopIndex },
// 初始化loopIndex =1 
    Expression.Assign(loopIndex, Expression.Constant(1)),
    Expression.Loop(
        Expression.IfThenElse(
            // if 的判斷邏輯
            Expression.LessThanOrEqual(loopIndex, Expression.Constant(10)),
            // 判斷邏輯通過的代碼
            Expression.Block(
                Expression.Call(
                    null,
                    typeof(Console).GetMethod("WriteLine", new Type[] { typeof(string) }),
                    Expression.Constant("Hello")),
                Expression.PostIncrementAssign(loopIndex)),
            // 判斷不通過的代碼
            Expression.Break(labelBreak)
            ),labelBreak));

// 將我們上面的代碼塊表達式
Expression<Action> lambdaExpression =  Expression.Lambda<Action>(block);
lambdaExpression.Compile().Invoke();

  

  希望上面的代碼沒有阻止你學習表達式樹的決心J 。

  好吧,我們又學了幾個新的類型的表達式,來總結一下:

  到這裡,我想大家應該對錶達式樹的構建有了一個清楚的認識。至於為什麼不允許我們直接基於複雜的Lambda表達式來創建表達式樹呢?

  • 這裡的Lambda表達式實際上是一個Expression Body。
  • 這個Expression Body實際上就是我們上面講到的Expression中的一種。
  • 也就是說編譯器需要時間去分析你到底是哪一種?
  • 最簡單的x=> x+1之類的也就是Func<TValue,TKey> 是很容易分析的。
  • 實際這裡面允許的Expression Body只有BinaryExpression。

  最後,我們來完整的看一下.NET都為我們提供了哪些類型的表達式(下麵這些類都是繼承自Expression)。

TypeBinaryExpression

TypeBinaryExpression typeBinaryExpression =
    Expression.TypeIs(
        Expression.Constant("spruce"),
        typeof(int));

Console.WriteLine(typeBinaryExpression.ToString());
// ("spruce" Is Int32)

IndexExpression

ParameterExpression arrayExpr = Expression.Parameter(typeof(int[]), "Array");

ParameterExpression indexExpr = Expression.Parameter(typeof(int), "Index");

ParameterExpression valueExpr = Expression.Parameter(typeof(int), "Value");

Expression arrayAccessExpr = Expression.ArrayAccess(
    arrayExpr,
    indexExpr
);

Expression<Func<int[], int, int, int>> lambdaExpr = Expression.Lambda<Func<int[], int, int, int>>(
        Expression.Assign(arrayAccessExpr, Expression.Add(arrayAccessExpr, valueExpr)),
        arrayExpr,
        indexExpr,
        valueExpr
    );

Console.WriteLine(arrayAccessExpr.ToString());
// Array[Index]

Console.WriteLine(lambdaExpr.ToString());
// (Array, Index, Value) => (Array[Index] = (Array[Index] + Value)) 

Console.WriteLine(lambdaExpr.Compile().Invoke(new int[] { 10, 20, 30 }, 0, 5));
// 15

NewExpression

NewExpression newDictionaryExpression =Expression.New(typeof(Dictionary<int, string>));
Console.WriteLine(newDictionaryExpression.ToString());
// new Dictionary`2()

InvocationExpression

Expression<Func<int, int, bool>> largeSumTest =
    (num1, num2) => (num1 + num2) > 1000;

InvocationExpression invocationExpression= Expression.Invoke(
    largeSumTest,
    Expression.Constant(539),
    Expression.Constant(281));

Console.WriteLine(invocationExpression.ToString());
// Invoke((num1, num2) => ((num1 + num2) > 1000),539,281)

  今天我們演示瞭如何通過代碼的方式去創建表達式樹,然後總結了一下.NET為我們提供的表達式類型。下一篇,我們將繼續研究表達式樹的遍歷問題,敬請期待,如果對於表達式樹有興趣的同學歡迎持續關註~,


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

-Advertisement-
Play Games
更多相關文章
  • 本篇文章雖不談架構,但是Cache又是架構中不可或缺的部分,因此,在講解Cache的同時,將會提及到部分架構知識,關於架構部分,讀者可以不用理解,或者直接跳過, 你只需關心Cache即可,具體的架構,會在後續文章中與大家分享。 一 為什麼要在ASP.NET 項目中引入緩存 1. 我們先來考慮一個問題 ...
  • 此系列文章圍繞著拼多多賣家工具來介紹ABPZero的使用,內容包括手機登錄、手機註冊、拼團提醒、微信公眾號綁定帳號、有拼團發送消息到微信公眾號(只要關註過微信公眾號並已綁定系統帳號)。 學習此系列必備: 手機驗證碼:使用阿裡雲簡訊,可註冊阿裡雲帳號訂購(後續會介紹) 微信公眾號:我所使用的是認證過的 ...
  • 第一種: string[] myArray = new string[10]; for(int i = 0; i<10;i++){ myArray[i] = i+1; } 第二種: string[] myArray2 = {"1","2","3","4","5","6","7","8","9","1 ...
  • 1.什麼是RESTful? REST,即Representational State Transfer的縮寫。"(資源的)表現層狀態轉化"。 2.什麼是表現層? "資源"具體呈現出來的形式,叫做它的"表現層"(Representation)。比如,文本可以用txt格式表現,也可以用HTML格式、XM ...
  • 創建響應式WinForm應用程式並不那麼簡單。 響應式佈局,我們在此指的是在不同屏幕解析度下的可用性。 對於WinForm應用程式,我們需要明確地根據解析度來調整控制項的大小和重新定位。 雖然在使用WPF時有相關的實踐應用,通過使用控制項的docking和anchoring,或使用panels等方法,但... ...
  • BlockingCollection集合是一個擁有阻塞功能的集合,它就是完成了經典生產者消費者的演算法功能。一般情況下,我們可以基於 生產者 - 消費者模式來實現併發。BlockingCollection<T> 類是最好的解決方案 剛結束的物聯網卡項目,我需要調用移動的某個具有批量獲取物聯網卡數據的接 ...
  • 概述 在之前寫的一篇關於async和await的前世今生的文章之後,大家似乎在async和await提高網站處理能力方面還有一些疑問,博客園本身也做了不少的嘗試。今天我們再來回答一下這個問題,同時我們會做一個async和await在WinForm中的嘗試,並且對比在4.5之前的非同步編程模式APM/E ...
  • 一個網頁,它是顯示圖片,但在一些瀏覽器,它卻顯示如下: Insus.NET猜,不是瀏覽器不相容,就是代碼有問題。 在代碼中,只是輸出數據流,圖片格式很多種,如jpg,png,bmp等,沒有指定,程式也不清楚要顯示什麼格式的圖片。因此,Insus.NET把代碼改為如下: context.Respons ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...