我试图生成一个"属性选择器"从一个字符串。
让我用一个真实的例子来解释一下自己:
我们有一个带有Name(string)属性的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;
}
但是......我得到了无效的施法错误!
无法隐式转换类型系统.Linq.Expressions.Expression&#39;至 &#39;&System.Linq.Expressions.Expression GT;&#39 ;.一个 存在显式转换(您是否错过了演员?)
我对表达式很新,我不知道如何继续:(
答案 0 :(得分:15)
你的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"