c# - 从字符串创建属性选择器表达式

标签 c# linq reflection lambda

我正在尝试从字符串生成“属性选择器”。

让我用一个现实生活中的例子来解释一下:

我们有一个带有 Name(字符串)属性的 Person 类。

我可以手动创建一个“属性选择器”,就像这样写 propertySelector:

Expression<Func<Person, string>> propertySelector = x => x.Name;

但我想用我的方法获得相同的属性选择器。

var propertySelector = CreatePropertySelectorExpression<Person, string>("Name");

我目前的情况是这样的:

public static Expression<Func<TIn, TOut>> CreatePropertySelectorExpression<TIn, TOut>(string path)
{
    Expression exp = Expression.Parameter(typeof(TIn), "x");
    foreach (var property in path.Split('.'))
    {
        exp = Expression.PropertyOrField(exp, property);
    }
    return exp;
}

但是...我遇到了无效的转换错误!

Cannot implicitly convert type 'System.Linq.Expressions.Expression' to 'System.Linq.Expressions.Expression>'. An explicit conversion exists (are you missing a cast?)

我是表达式的新手,我不知道如何继续:(

最佳答案

您的 exp 只包含 lambda 的主体。但是你想要一个实际的 lambda 函数,它在那里接受 TIn 类型的参数。因此,您需要使用 Expression.Lambda 创建一个 lambda:

var param = Expression.Parameter(typeof(TIn));
var body = Expression.PropertyOrField(param, propertyName);
return Expression.Lambda<Func<TIn, TOut>>(body, param);

请注意,该表达式并不能真正帮助您。您可能需要一个编译函数:

private static Func<TIn, TOut> CreatePropertyAccessor<TIn, TOut> (string propertyName)
{
    var param = Expression.Parameter(typeof(TIn));
    var body = Expression.PropertyOrField(param, propertyName);
    return Expression.Lambda<Func<TIn, TOut>>(body, param).Compile();
}

然后你可以像这样使用它:

var name1 = CreatePropertyAccessor<Obj, string>("Name");
var name2 = CreatePropertyAccessor<Obj, string>("Name2");
var name3 = CreatePropertyAccessor<Obj, string>("Name3");

var o = new Obj() // Obj is a type with those three properties
{
    Name = "foo",
    Name2 = "bar",
    Name3 = "baz"
};

Console.WriteLine(name1(o)); // "foo"
Console.WriteLine(name2(o)); // "bar"
Console.WriteLine(name3(o)); // "baz"

关于c# - 从字符串创建属性选择器表达式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27669993/

相关文章:

java - 通过反射初始化字符串常量

c# - 在 C# 中将字符串转换为日期

c# - 检测 LambdaExpression 是否用于对象的属性

c# - 在 C# 中调用多个 "SQL DataReader"的正确方法是什么?

c# - 当 T 为 IEnumerable<TP> 时,将类型 T 的泛型参数转换为 List<TP>

Java 反射 : How to obtain an instance of Class<T> from the ParameterizedType. getActualTypeArguments() 数组?

c# - 使用 CaSTLe windsor 创建对象而不是工厂类

c# - 如何检测不同的用户设置?启用 Javascript,启用 Cookie

c# - 无法将 lambda 表达式转换为类型 'string',因为它不是委托(delegate)类型

c# - 使用 linq 从字符串中提取对象列表