我正在尝试制作一个用于打印标签的简单脚本系统。我在过去做过这个没有问题的反射,但我现在正在尝试使用Lambda函数,以便我可以缓存函数以便重用。
我到目前为止的代码如下......
public static string GetValue<T>(T source, string propertyPath) {
try {
Func<T, Object> func;
Type type = typeof(T);
ParameterExpression parameterExpression = Expression.Parameter(type, @"source");
Expression expression = parameterExpression;
foreach (string property in propertyPath.Split('.')) {
PropertyInfo propertyInfo = type.GetProperty(property);
expression = Expression.Property(expression, propertyInfo);
type = propertyInfo.PropertyType;
}
func = Expression.Lambda<Func<T, Object>>(expression, parameterExpression).Compile();
object value = func.Invoke(source);
if (value == null)
return string.Empty;
return value.ToString();
}
catch {
return propertyPath;
}
}
这似乎在某些情况下有效,但在其他情况下却失败了。问题似乎在于我试图将值作为对象返回 - 无论实际数据类型如何。我试图这样做,因为我不知道在编译时数据类型是什么,但从长远来看,我只需要一个字符串。
每当我尝试访问Int32类型的属性时,我都会收到此消息标题中显示的异常 - 但我也是为Nullable类型和其他类型获取它。 当我尝试将表达式编译到函数中时抛出异常。
有人可以建议我在保持Lambda功能的同时以不同的方式解决这个问题,以便我可以缓存访问者吗?
答案 0 :(得分:101)
您是否尝试过使用Expression.Convert?这将增加拳击/提升/等转换。
Expression conversion = Expression.Convert(expression, typeof(object));
func = Expression.Lambda<Func<T, Object>>(conversion, parameterExpression).Compile();
答案 1 :(得分:0)
我希望这段代码对您有所帮助
using System;
using System.Linq;
using System.Linq.Expressions;
using System.Reflection;
namespace Student
{
class Program
{
static void Main(string[] args)
{
var a = new Student();
PrintProperty(a, "Name");
PrintProperty(a, "Age");
Console.ReadKey();
}
private static void PrintProperty<T>(T a, string propName)
{
PrintProperty<T, object>(a, propName);
}
private static void PrintProperty<T, TProperty>(T a, string propName)
{
ParameterExpression ep = Expression.Parameter(typeof(T), "x");
MemberExpression em = Expression.Property(ep, typeof(T).GetProperty(propName));
var el = Expression.Lambda<Func<T, TProperty>>(Expression.Convert(em, typeof(object)), ep);
Console.WriteLine(GetValue(a, el));
}
private static TPorperty GetValue<T, TPorperty>(T v, Expression<Func<T, TPorperty>> expression)
{
return expression.Compile().Invoke(v);
}
public class Student
{
public Student()
{
Name = "Albert Einstein";
Age = 15;
}
public string Name { get; set; }
public int Age { get; set; }
}
}
}