在C# .NET中,Func、Predicate和Expression是三种常用的委托和表达式类型,它们在编写灵活、可重用的代码时非常有用。本文将详细介绍这三种类型,并提供多个实例来说明它们的用法和区别。
1. Func<T, TResult>
Func是一个通用委托,它可以接受零个或多个输入参数,并返回一个值。其基本形式为:
public delegate TResult Func<out TResult>();
public delegate TResult Func<in T, out TResult>(T arg);
public delegate TResult Func<in T1, in T2, out TResult>(T1 arg1, T2 arg2);
// ... 最多可以有16个输入参数
示例1:基本用法
Func<int, int, string> formatNumber = (a, b) => $"The sum of {a} and {b} is {a + b}";
string result = formatNumber(5, 3);
Console.WriteLine(result);
示例2:作为方法参数
public static List<int> FilterList(List<int> numbers, Func<int, bool> filterFunc)
{
return numbers.Where(filterFunc).ToList();
}
static void Main(string[] args)
{
// 使用
List<int> numbers = new List<int> { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
List<int> evenNumbers = FilterList(numbers, n => n % 2 == 0);
Console.WriteLine(string.Join(", ", evenNumbers));
}
2. Predicate<T>
Predicate是一个特殊的Func,它始终返回一个布尔值。它通常用于定义过滤条件。
public delegate bool Predicate<in T>(T obj);
示例3:使用Predicate
static void Main(string[] args)
{
List<string> fruits = new List<string> { "apple", "banana", "cherry", "date", "elderberry" };
Predicate<string> startsWithB = s => s.StartsWith("b", StringComparison.OrdinalIgnoreCase);
string bFruit = fruits.Find(startsWithB);
Console.WriteLine(bFruit);
}
示例4:Predicate vs Func
static void Main(string[] args)
{
// 使用Predicate
Predicate<int> isEven = n => n % 2 == 0;
List<int> numbers = new List<int> { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
List<int> evenNumbers = numbers.FindAll(isEven);
// 使用Func
Func<int, bool> isOdd = n => n % 2 != 0;
List<int> oddNumbers = numbers.Where(isOdd).ToList();
Console.WriteLine($"Even numbers: {string.Join(", ", evenNumbers)}");
Console.WriteLine($"Odd numbers: {string.Join(", ", oddNumbers)}");
}
3. Expression<T>
Expression表示一个可以编译和执行的代码块。它们通常用于构建动态查询、规则引擎或者在运行时修改代码行为。
示例5:基本Expression
static void Main(string[] args)
{
Expression<Func<int, bool>> isPositive = n => n > 0;
// 编译并执行Expression
Func<int, bool> compiledFunc = isPositive.Compile();
bool result = compiledFunc(5);
Console.WriteLine(result);
}
示例6:构建动态查询
public class Person
{
public string Name { get; set; }
public int Age { get; set; }
}
public static Expression<Func<Person, bool>> BuildAgeRangeExpression(int minAge, int maxAge)
{
ParameterExpression parameter = Expression.Parameter(typeof(Person), "p");
Expression left = Expression.Property(parameter, "Age");
Expression minAgeCheck = Expression.GreaterThanOrEqual(left, Expression.Constant(minAge));
Expression maxAgeCheck = Expression.LessThanOrEqual(left, Expression.Constant(maxAge));
Expression combinedCheck = Expression.AndAlso(minAgeCheck, maxAgeCheck);
return Expression.Lambda<Func<Person, bool>>(combinedCheck, parameter);
}
static void Main(string[] args)
{
// 使用
List<Person> people = new List<Person>
{
new Person { Name = "Alice", Age = 25 },
new Person { Name = "Bob", Age = 30 },
new Person { Name = "Charlie", Age = 35 },
new Person { Name = "David", Age = 40 }
};
var ageRangeExpression = BuildAgeRangeExpression(28, 38);
var filteredPeople = people.AsQueryable().Where(ageRangeExpression);
foreach (var person in filteredPeople)
{
Console.WriteLine($"{person.Name} - {person.Age}");
}
}
总结
Func<T, TResult> 是一个通用委托,可以接受多个输入参数并返回一个值。它非常灵活,适用于多种场景。
Predicate<T> 是Func的一个特例,专门用于返回布尔值的情况。它通常用于定义过滤条件。
Expression<T> 表示可编译和执行的代码块。它允许在运行时检查、修改和编译代码,特别适用于构建动态查询和规则引擎。
这三种类型在C# .NET编程中扮演着重要角色,能够帮助开发者编写更加灵活、可重用和高效的代码。根据具体的使用场景,选择合适的类型可以大大提高代码的表现力和可维护性。
该文章在 2024/11/7 10:27:20 编辑过