C#表達(dá)式樹(shù)(Expression Trees)的使用
什么是表達(dá)式樹(shù)?
- 表達(dá)式樹(shù)是C#中一種數(shù)據(jù)結(jié)構(gòu),用于以樹(shù)狀方式表示代碼中的表達(dá)式,每個(gè)節(jié)點(diǎn)代表一個(gè)操作(如算術(shù)運(yùn)算、方法調(diào)用等)。
- 它們?cè)试S你將代碼本身視為數(shù)據(jù)結(jié)構(gòu),能夠在運(yùn)行時(shí)動(dòng)態(tài)地分析、修改和執(zhí)行代碼。
- 表達(dá)式樹(shù)最初是在 .NET 3.5 中引入的,主要用于支持 LINQ(語(yǔ)言集成查詢)。
核心概念
1.表達(dá)式樹(shù)的構(gòu)建
- 表達(dá)式樹(shù)的核心類型位于 System.Linq.Expressions 命名空間。
- 可以手動(dòng)構(gòu)建表達(dá)式樹(shù),也可以通過(guò)Lambda表達(dá)式隱式構(gòu)建。
using System; using System.Linq.Expressions; class Program { static void Main() { // 創(chuàng)建一個(gè)簡(jiǎn)單的表達(dá)式:x => x + 1 ParameterExpression param = Expression.Parameter(typeof(int), "x"); BinaryExpression body = Expression.Add(param, Expression.Constant(1)); Expression<Func<int, int>> expression = Expression.Lambda<Func<int, int>>(body, param); // 編譯并執(zhí)行表達(dá)式樹(shù) Func<int, int> compiledExpression = expression.Compile(); int result = compiledExpression(5); Console.WriteLine($"Result: {result}"); // 輸出:Result: 6 } }
在這個(gè)示例中,我們創(chuàng)建了一個(gè)簡(jiǎn)單的表達(dá)式樹(shù)表示 x => x + 1,并將其編譯成可執(zhí)行代碼。
2. 表達(dá)式樹(shù)與Lambda表達(dá)式
Lambda表達(dá)式可以被編譯為委托,也可以被表達(dá)式樹(shù)捕獲:
Expression<Func<int, int>> square = x => x * x;
此時(shí),square不是一個(gè)委托,而是一棵描述 x * x 計(jì)算過(guò)程的樹(shù),可用于分析和轉(zhuǎn)換。
3.解析和訪問(wèn)表達(dá)式樹(shù)
表達(dá)式樹(shù)可以遍歷并分析其結(jié)構(gòu):
void PrintExpression(Expression exp, int level = 0) { Console.WriteLine(new string(' ', level * 2) + exp.NodeType + " - " + exp.Type); if (exp is BinaryExpression bin) { PrintExpression(bin.Left, level + 1); PrintExpression(bin.Right, level + 1); } else if (exp is ParameterExpression param) { Console.WriteLine(new string(' ', (level+1) * 2) + "Parameter: " + param.Name); } }
4.動(dòng)態(tài)條件查詢
我們有一個(gè) Product 類和一個(gè)產(chǎn)品列表。我們希望根據(jù)產(chǎn)品的價(jià)格動(dòng)態(tài)過(guò)濾產(chǎn)品。
using System; using System.Collections.Generic; using System.Linq; using System.Linq.Expressions; public class Product { public string Name { get; set; } public decimal Price { get; set; } } class Program { static void Main() { // 創(chuàng)建產(chǎn)品列表 List<Product> products = new List<Product> { new Product { Name = "Laptop", Price = 1000m }, new Product { Name = "Smartphone", Price = 500m }, new Product { Name = "Tablet", Price = 300m } }; // 動(dòng)態(tài)創(chuàng)建表達(dá)式樹(shù)來(lái)過(guò)濾價(jià)格大于 400 的產(chǎn)品 Func<Product, bool> filter = CreatePriceFilter(400m); // 使用生成的過(guò)濾器查詢產(chǎn)品 var filteredProducts = products.Where(filter).ToList(); foreach (var product in filteredProducts) { Console.WriteLine($"Product: {product.Name}, Price: {product.Price}"); } } static Func<Product, bool> CreatePriceFilter(decimal minPrice) { // 創(chuàng)建參數(shù)表達(dá)式 ParameterExpression param = Expression.Parameter(typeof(Product), "product"); // 創(chuàng)建訪問(wèn)屬性表達(dá)式 MemberExpression priceProperty = Expression.Property(param, "Price"); // 創(chuàng)建常量表達(dá)式 ConstantExpression constant = Expression.Constant(minPrice); // 創(chuàng)建大于運(yùn)算符表達(dá)式 BinaryExpression comparison = Expression.GreaterThan(priceProperty, constant); // 創(chuàng)建 lambda 表達(dá)式 Expression<Func<Product, bool>> lambda = Expression.Lambda<Func<Product, bool>>(comparison, param); // 編譯表達(dá)式樹(shù)為可執(zhí)行代碼 return lambda.Compile(); } }
1.設(shè)置產(chǎn)品列表:
- 我們先定義一個(gè)簡(jiǎn)單的 Product 類和一個(gè)包含幾個(gè)產(chǎn)品的列表。
2.創(chuàng)建表達(dá)式樹(shù):
- 參數(shù)表達(dá)式:ParameterExpression param = Expression.Parameter(typeof(Product), "product"); 創(chuàng)建一個(gè)參數(shù),表示傳遞給過(guò)濾器的 Product 對(duì)象。
- 屬性訪問(wèn)表達(dá)式:MemberExpression priceProperty = Expression.Property(param, "Price"); 訪問(wèn)傳遞對(duì)象的 Price 屬性。
- 常量表達(dá)式:ConstantExpression constant = Expression.Constant(minPrice); 定義過(guò)濾條件中的常量值。
- 比較表達(dá)式:BinaryExpression comparison = Expression.GreaterThan(priceProperty, constant); 創(chuàng)建一個(gè)比較表達(dá)式,檢查 Price 是否大于 minPrice。
- lambda 表達(dá)式:將上述表達(dá)式組合成一個(gè)完整的 lambda 表達(dá)式,并編譯成可執(zhí)行代碼。
3.應(yīng)用表達(dá)式:
- 使用 Where 方法將生成的過(guò)濾器應(yīng)用于產(chǎn)品列表,并輸出結(jié)果。
表達(dá)式樹(shù)的優(yōu)勢(shì)
1.動(dòng)態(tài)構(gòu)建查詢
- 表達(dá)式樹(shù)允許你在運(yùn)行時(shí)構(gòu)建和修改查詢邏輯。這在需要根據(jù)用戶輸入或其他動(dòng)態(tài)數(shù)據(jù)生成不同查詢條件時(shí)特別有用。
2.LINQ 提供程序支持:
- 表達(dá)式樹(shù)是 LINQ 提供程序(如 LINQ to SQL、Entity Framework)的基礎(chǔ),它們將表達(dá)式樹(shù)解析為底層數(shù)據(jù)源(如數(shù)據(jù)庫(kù)、XML)的查詢語(yǔ)言。這意味著你可以用相同的代碼生成運(yùn)行在不同數(shù)據(jù)源上的查詢。
3.性能優(yōu)化
- 在某些情況下,表達(dá)式樹(shù)可以被編譯和緩存,提高重復(fù)執(zhí)行相同邏輯的性能。
4.元數(shù)據(jù)處理
- 表達(dá)式樹(shù)提供對(duì)表達(dá)式結(jié)構(gòu)的訪問(wèn),這使得分析和處理代碼元數(shù)據(jù)成為可能。這對(duì)于開(kāi)發(fā)動(dòng)態(tài)應(yīng)用程序或框架尤其有用。
5.代碼轉(zhuǎn)換和重寫(xiě)
- 可以編寫(xiě)代碼來(lái)遍歷和修改表達(dá)式樹(shù),用于實(shí)現(xiàn)代碼轉(zhuǎn)換或重寫(xiě)。這對(duì)于構(gòu)建復(fù)雜查詢或分析工具有很大幫助。
適用場(chǎng)景
- 動(dòng)態(tài)條件查詢:當(dāng)應(yīng)用需要支持用戶定義的動(dòng)態(tài)過(guò)濾條件時(shí),表達(dá)式樹(shù)可以靈活地構(gòu)建這些條件。
- 跨平臺(tái)查詢:在 LINQ to SQL 或 Entity Framework 中,表達(dá)式樹(shù)可被翻譯成 SQL 查詢,在數(shù)據(jù)庫(kù)執(zhí)行。
- 規(guī)則引擎和DSL(領(lǐng)域特定語(yǔ)言):在這些場(chǎng)景中,表達(dá)式樹(shù)可以用于解析和執(zhí)行用戶定義的規(guī)則或查詢
代碼復(fù)雜性的權(quán)衡
- 雖然表達(dá)式樹(shù)代碼在某些情況下顯得復(fù)雜,但其提供的靈活性和功能在復(fù)雜應(yīng)用中是非常關(guān)鍵的。
- 如果只是簡(jiǎn)單的篩選條件,直接使用 Lambda 表達(dá)式或 LINQ 查詢語(yǔ)法更為直接和清晰。
示例1:
假設(shè)我們有一個(gè)產(chǎn)品列表,用戶可以動(dòng)態(tài)選擇多個(gè)條件進(jìn)行過(guò)濾,比如根據(jù)名稱、價(jià)格范圍或庫(kù)存狀態(tài)等進(jìn)行篩選。我們需要在運(yùn)行時(shí)根據(jù)用戶輸入組合這些條件。
using System; using System.Collections.Generic; using System.Linq; using System.Linq.Expressions; public class Product { public string Name { get; set; } public decimal Price { get; set; } public bool InStock { get; set; } } class Program { static void Main() { var products = new List<Product> { new Product { Name = "Laptop", Price = 1000, InStock = true }, new Product { Name = "Smartphone", Price = 500, InStock = true }, new Product { Name = "Tablet", Price = 300, InStock = false } }; // 用戶可以選擇動(dòng)態(tài)條件 string searchName = "Laptop"; decimal? minPrice = 400; decimal? maxPrice = null; bool? inStock = true; // 創(chuàng)建動(dòng)態(tài)查詢表達(dá)式 var filter = CreateDynamicFilter<Product>(searchName, minPrice, maxPrice, inStock); // 使用生成的過(guò)濾器查詢產(chǎn)品 var filteredProducts = products.AsQueryable().Where(filter).ToList(); foreach (var product in filteredProducts) { Console.WriteLine($"Product: {product.Name}, Price: {product.Price}, InStock: {product.InStock}"); } } static Expression<Func<T, bool>> CreateDynamicFilter<T>(string name, decimal? minPrice, decimal? maxPrice, bool? inStock) { // 參數(shù)表達(dá)式 var parameter = Expression.Parameter(typeof(T), "product"); Expression expression = Expression.Constant(true); // 初始謂詞為 true // 根據(jù) name 動(dòng)態(tài)創(chuàng)建條件 if (!string.IsNullOrEmpty(name)) { var nameProperty = Expression.Property(parameter, "Name"); var nameValue = Expression.Constant(name); var nameExpression = Expression.Equal(nameProperty, nameValue); expression = Expression.AndAlso(expression, nameExpression); } // 根據(jù) minPrice 創(chuàng)建條件 if (minPrice.HasValue) { var priceProperty = Expression.Property(parameter, "Price"); var minPriceValue = Expression.Constant(minPrice.Value); var minPriceExpression = Expression.GreaterThanOrEqual(priceProperty, minPriceValue); expression = Expression.AndAlso(expression, minPriceExpression); } // 根據(jù) maxPrice 創(chuàng)建條件 if (maxPrice.HasValue) { var priceProperty = Expression.Property(parameter, "Price"); var maxPriceValue = Expression.Constant(maxPrice.Value); var maxPriceExpression = Expression.LessThanOrEqual(priceProperty, maxPriceValue); expression = Expression.AndAlso(expression, maxPriceExpression); } // 根據(jù) inStock 創(chuàng)建條件 if (inStock.HasValue) { var stockProperty = Expression.Property(parameter, "InStock"); var stockValue = Expression.Constant(inStock.Value); var stockExpression = Expression.Equal(stockProperty, stockValue); expression = Expression.AndAlso(expression, stockExpression); } // 創(chuàng)建 Lambda 表達(dá)式 return Expression.Lambda<Func<T, bool>>(expression, parameter); } }
示例2:
針對(duì)上文中只針對(duì)價(jià)格做篩選的示例,那么我們的篩選過(guò)程完全可以簡(jiǎn)化成如下表達(dá)式
var filteredProducts = products.Where(p => p.Price > 400).ToList();
總結(jié)來(lái)說(shuō),是否使用表達(dá)式樹(shù)取決于你的具體需求和應(yīng)用場(chǎng)景。在需要?jiǎng)討B(tài)處理和復(fù)雜邏輯的情況下,表達(dá)式樹(shù)提供了強(qiáng)大的工具支持,而在簡(jiǎn)單場(chǎng)景下,直接使用 Lambda 表達(dá)式或常規(guī)方法更為合適。希望這能幫助你理解表達(dá)式樹(shù)的適用場(chǎng)景和優(yōu)勢(shì)!
到此這篇關(guān)于C#表達(dá)式樹(shù)(Expression Trees)的使用的文章就介紹到這了,更多相關(guān)C#表達(dá)式樹(shù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- C#之Expression表達(dá)式樹(shù)實(shí)例
- C# 表達(dá)式樹(shù)Expression Trees的知識(shí)梳理
- C#表達(dá)式樹(shù)Expression動(dòng)態(tài)創(chuàng)建表達(dá)式
- 淺談c#表達(dá)式樹(shù)Expression簡(jiǎn)單類型比較demo
- C#表達(dá)式樹(shù)Expression基礎(chǔ)講解
- C#表達(dá)式樹(shù)的基本用法講解
- C#表達(dá)式樹(shù)講解
- C#表達(dá)式樹(shù)基礎(chǔ)教程
- C#執(zhí)行表達(dá)式樹(shù)(Expression Tree)的具體使用
相關(guān)文章
C#中System.Text.Json匿名對(duì)象反序列化
這篇文章主要介紹了System.Text.Json匿名對(duì)象反序列化,下文代碼基于. NET 6,為了代碼整潔,實(shí)際配置了PropertyNameCaseInsensitive = true,本文結(jié)合實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-05-05C#中數(shù)組、ArrayList、List、Dictionary的用法與區(qū)別淺析(存取數(shù)據(jù))
在工作中經(jīng)常遇到C#數(shù)組、ArrayList、List、Dictionary存取數(shù)據(jù),但是該選擇哪種類型進(jìn)行存儲(chǔ)數(shù)據(jù)呢?很迷茫,今天小編抽空給大家整理下這方面的內(nèi)容,需要的朋友參考下吧2017-02-02C#實(shí)現(xiàn)汽車(chē)租賃系統(tǒng)項(xiàng)目
這篇文章主要為大家詳細(xì)介紹了C#實(shí)現(xiàn)汽車(chē)租賃系統(tǒng)項(xiàng)目,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-01-01利用lambda表達(dá)式樹(shù)優(yōu)化反射詳解
這篇文章主要給大家介紹了關(guān)于如何利用lambda表達(dá)式樹(shù)優(yōu)化反射的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2018-12-12C#實(shí)現(xiàn)簡(jiǎn)單的Login窗口實(shí)例
這篇文章主要介紹了C#實(shí)現(xiàn)簡(jiǎn)單的Login窗口,實(shí)例分析了C#顯示及關(guān)閉登陸Login窗口的技巧,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2015-08-08C#實(shí)現(xiàn)計(jì)算年齡的簡(jiǎn)單方法匯總
本文給大家分享的是C#代碼實(shí)現(xiàn)的簡(jiǎn)單實(shí)用的給出用戶的出生日期,計(jì)算出用戶的年齡的代碼,另外附上其他網(wǎng)友的方法,算是對(duì)計(jì)算年齡的一次小結(jié),希望大家能夠喜歡。2015-05-05換個(gè)方式使用C#開(kāi)發(fā)微信小程序的過(guò)程
這篇文章主要介紹了換個(gè)方式使用C#開(kāi)發(fā)微信小程序的過(guò)程,演示使用C#寫(xiě)的LiveCharts,點(diǎn)擊按鈕動(dòng)態(tài)生成一些數(shù)據(jù),感興趣的朋友跟隨小編一起看看吧2025-05-05C#中圖片.BYTE[]和base64string的轉(zhuǎn)換方法
下面小編就為大家?guī)?lái)一篇C#中圖片.BYTE[]和base64string的轉(zhuǎn)換方法。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-02-02