首页 > 解决方案 > C# 反射表达式 Linq

问题描述

我在使用 C# 动态生成的 lambda 表达式时遇到了一些问题。

考虑以下场景:

public class Person {
    public long Id { get; set; }
    public string Name { get; set; }
}

List<Person> persons = new List<Person> () {
    new Person { Id = 1, Name = "Foo" },
    new Person { Id = 2, Name = "Bar" },
    new Person { Id = 3, Name = "Baz" },
    new Person { Id = 4, Name = null },
};

现在,执行以下代码

ParameterExpression param = Expression.Parameter(typeof(Person), "arg");
Expression prop = Expression.Property(param, "Name");
Expression value = Expression.Constant("bar");
Type type = prop.Type;

MethodInfo toLower = typeof(String).GetMethod("ToLower", Type.EmptyTypes);
Expression expLower = Expression.Call(prop, toLower);

Expression clausule = Expression.Call(expLower, type.GetMethod("Contains", new[] { type }), value);
Expression notNull = Expression.NotEqual(prop, Expression.Constant(null));

clausule = Expression.And(notNull, clausule);

var exp = Expression.Lambda<Func<T, bool>>(clausule, param);

上面的代码生成下面的exp。

//arg => ((arg.Name != null) And (arg.Name.ToLower().Contains("bar")))

现在,尝试将其应用到我的列表中。

下面的过滤器有效

var filteredListThatWorks = persons.Where(arg => arg.Name != null && arg.Name.ToLower().Contains("bar")).ToList();

下面的一个抛出 Null 对象的异常(因为 Id 4 名称)

var filteredListThatGivesExp = persons.Where(exp.Compile()).ToList();

同样的表达式,当由 lambda 生成时,会抛出 exp,当手动输入时,它可以工作。任何人都知道解决这个问题的方法吗?

溴,

标签: c#lambdareflection

解决方案


And&;你想使用AndAlso&&):

clausule = Expression.AndAlso(notNull, clausule);

如果有疑问,sharplab.io 是一个很好的工具;如果我使用:

Expression<Func<Person, bool>> filter
        = arg => arg.Name != null && arg.Name.ToLower().Contains("bar");

它告诉我它编译为相当于:

// ...
BinaryExpression body = Expression.AndAlso(left, Expression.Call(instance, method, obj));
// ...

(请注意,它必须对某些指令撒谎,因为它编译为实际上无法用原始 C# 表达的东西)

看到它在行动


推荐阅读