C#学习第29天:表达式树(Expression Trees)

目录

什么是表达式树?

 核心概念

1.表达式树的构建

2. 表达式树与Lambda表达式 

3.解析和访问表达式树

4.动态条件查询

表达式树的优势

1.动态构建查询

2.LINQ 提供程序支持:

3.性能优化

4.元数据处理

5.代码转换和重写

适用场景 

代码复杂性的权衡 


什么是表达式树?


  • 表达式树是C#中一种数据结构,用于以树状方式表示代码中的表达式,每个节点代表一个操作(如算术运算、方法调用等)。
  • 它们允许你将代码本身视为数据结构,能够在运行时动态地分析、修改和执行代码。
  • 表达式树最初是在 .NET 3.5 中引入的,主要用于支持 LINQ(语言集成查询)。

 核心概念


1.表达式树的构建

  • 表达式树的核心类型位于 System.Linq.Expressions 命名空间。
  • 可以手动构建表达式树,也可以通过Lambda表达式隐式构建。
using System;
using System.Linq.Expressions;

class Program
{
    static void Main()
    {
        // 创建一个简单的表达式: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);

        // 编译并执行表达式树
        Func<int, int> compiledExpression = expression.Compile();
        int result = compiledExpression(5);

        Console.WriteLine($"Result: {result}"); // 输出:Result: 6
    }
}

 在这个示例中,我们创建了一个简单的表达式树表示 x => x + 1,并将其编译成可执行代码。

2. 表达式树与Lambda表达式 

Lambda表达式可以被编译为委托,也可以被表达式树捕获:

Expression<Func<int, int>> square = x => x * x;

此时,square不是一个委托,而是一棵描述 x * x 计算过程的树,可用于分析和转换。

3.解析和访问表达式树

表达式树可以遍历并分析其结构:

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.动态条件查询

我们有一个 Product 类和一个产品列表。我们希望根据产品的价格动态过滤产品。

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()
    {
        // 创建产品列表
        List<Product> products = new List<Product>
        {
            new Product { Name = "Laptop", Price = 1000m },
            new Product { Name = "Smartphone", Price = 500m },
            new Product { Name = "Tablet", Price = 300m }
        };

        // 动态创建表达式树来过滤价格大于 400 的产品
        Func<Product, bool> filter = CreatePriceFilter(400m);

        // 使用生成的过滤器查询产品
        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)
    {
        // 创建参数表达式
        ParameterExpression param = Expression.Parameter(typeof(Product), "product");

        // 创建访问属性表达式
        MemberExpression priceProperty = Expression.Property(param, "Price");

        // 创建常量表达式
        ConstantExpression constant = Expression.Constant(minPrice);

        // 创建大于运算符表达式
        BinaryExpression comparison = Expression.GreaterThan(priceProperty, constant);

        // 创建 lambda 表达式
        Expression<Func<Product, bool>> lambda = Expression.Lambda<Func<Product, bool>>(comparison, param);

        // 编译表达式树为可执行代码
        return lambda.Compile();
    }
}

1.设置产品列表:

  •  我们先定义一个简单的 Product 类和一个包含几个产品的列表。

2.创建表达式树:

  • 参数表达式:ParameterExpression param = Expression.Parameter(typeof(Product), "product"); 创建一个参数,表示传递给过滤器的 Product 对象。
  • 属性访问表达式:MemberExpression priceProperty = Expression.Property(param, "Price"); 访问传递对象的 Price 属性。
  • 常量表达式:ConstantExpression constant = Expression.Constant(minPrice); 定义过滤条件中的常量值。
  • 比较表达式:BinaryExpression comparison = Expression.GreaterThan(priceProperty, constant); 创建一个比较表达式,检查 Price 是否大于 minPrice。
  • lambda 表达式:将上述表达式组合成一个完整的 lambda 表达式,并编译成可执行代码。

3.应用表达式:

  • 使用 Where 方法将生成的过滤器应用于产品列表,并输出结果。

表达式树的优势


1.动态构建查询

  • 表达式树允许你在运行时构建和修改查询逻辑。这在需要根据用户输入或其他动态数据生成不同查询条件时特别有用。

2.LINQ 提供程序支持:

  • 表达式树是 LINQ 提供程序(如 LINQ to SQL、Entity Framework)的基础,它们将表达式树解析为底层数据源(如数据库、XML)的查询语言。这意味着你可以用相同的代码生成运行在不同数据源上的查询。

3.性能优化

  • 在某些情况下,表达式树可以被编译和缓存,提高重复执行相同逻辑的性能。

4.元数据处理

  • 表达式树提供对表达式结构的访问,这使得分析和处理代码元数据成为可能。这对于开发动态应用程序或框架尤其有用。

5.代码转换和重写

  • 可以编写代码来遍历和修改表达式树,用于实现代码转换或重写。这对于构建复杂查询或分析工具有很大帮助。

适用场景 


  • 动态条件查询:当应用需要支持用户定义的动态过滤条件时,表达式树可以灵活地构建这些条件。
  • 跨平台查询:在 LINQ to SQL 或 Entity Framework 中,表达式树可被翻译成 SQL 查询,在数据库执行。
  • 规则引擎和DSL(领域特定语言):在这些场景中,表达式树可以用于解析和执行用户定义的规则或查询 

代码复杂性的权衡 


  • 虽然表达式树代码在某些情况下显得复杂,但其提供的灵活性和功能在复杂应用中是非常关键的。
  • 如果只是简单的筛选条件,直接使用 Lambda 表达式或 LINQ 查询语法更为直接和清晰。

示例1:

假设我们有一个产品列表,用户可以动态选择多个条件进行过滤,比如根据名称、价格范围或库存状态等进行筛选。我们需要在运行时根据用户输入组合这些条件。

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 }
        };

        // 用户可以选择动态条件
        string searchName = "Laptop";
        decimal? minPrice = 400;
        decimal? maxPrice = null;
        bool? inStock = true;

        // 创建动态查询表达式
        var filter = CreateDynamicFilter<Product>(searchName, minPrice, maxPrice, inStock);

        // 使用生成的过滤器查询产品
        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)
    {
        // 参数表达式
        var parameter = Expression.Parameter(typeof(T), "product");
        Expression expression = Expression.Constant(true); // 初始谓词为 true

        // 根据 name 动态创建条件
        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);
        }

        // 根据 minPrice 创建条件
        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);
        }

        // 根据 maxPrice 创建条件
        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);
        }

        // 根据 inStock 创建条件
        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);
        }

        // 创建 Lambda 表达式
        return Expression.Lambda<Func<T, bool>>(expression, parameter);
    }
}

示例2:

针对上文中只针对价格做筛选的示例,那么我们的筛选过程完全可以简化成如下表达式

var filteredProducts = products.Where(p => p.Price > 400).ToList();

总结来说,是否使用表达式树取决于你的具体需求和应用场景。在需要动态处理和复杂逻辑的情况下,表达式树提供了强大的工具支持,而在简单场景下,直接使用 Lambda 表达式或常规方法更为合适。希望这能帮助你理解表达式树的适用场景和优势!

如果你有其他问题或需要进一步的帮助,请随时告诉我。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ghost143

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值