“System.Int32”类型的 ParameterExpression 不能用于“System.String”类型的委托参数

kan*_*ans 4 c# generics delegates dynamic expression-trees

我正在使用表达式树创建一个通用的 setter,这是我的代码

public Expression<Action<T,string>> GetAction<T>(string fieldName)
{
    ParameterExpression targetExpr = Expression.Parameter(typeof(T), "Target");  
    MemberExpression fieldExpr = Expression.Property(targetExpr, fieldName);    
    ParameterExpression valueExpr = Expression.Parameter(fieldExpr.Type, "value"); 
    UnaryExpression valueCast = (!fieldExpr.Type.IsValueType) 
                              ? Expression.TypeAs(valueExpr, fieldExpr.Type) 
                              : Expression.Convert(valueExpr, fieldExpr.Type);
    BinaryExpression assignExpr = Expression.Assign(fieldExpr, valueCast);    
    return Expression.Lambda<Action<T, string>>(assignExpr, targetExpr, valueExpr);
}
Run Code Online (Sandbox Code Playgroud)

我不调用.Compile()上述方法,因为我想检查它构建的表达式。

我的目标是

class Person
{
    public string FirstName { get; set; }
    public string LastName { get; set; }
    public int Age { get; set; }

}
Run Code Online (Sandbox Code Playgroud)

我这样调用方法

var lastname = GetAction<Person>("FirstName");
var age = GetAction<Person>("Age");

lastname.Compile()(p, "Solutions");
age.Compile()(p, "10");
Run Code Online (Sandbox Code Playgroud)

我将年龄作为字符串传递的原因是,我将从 XML 获取此值。

它正在为FirstName没有任何错误而创建 Action而为Age它吹。

错误发生在这一行Age

 return Expression.Lambda<Action<T, string>>(assignExpr, targetExpr, valueExpr);
Run Code Online (Sandbox Code Playgroud)

错误:

“System.Int32”类型的 ParameterExpression 不能用于“System.String”类型的委托参数

我可以用动态做点什么吗?

我希望有人会有一些解决方案。谢谢

Bal*_*nyi 5

您应该调用Convert.ChangeType类型转换:

public static Expression<Action<T, string>> GetAction<T>(string fieldName)
{
    ParameterExpression targetExpr = Expression.Parameter(typeof(T), "Target");
    MemberExpression fieldExpr = Expression.Property(targetExpr, fieldName);
    ParameterExpression valueExpr = Expression.Parameter(typeof(string), "value");
    MethodCallExpression convertExpr = Expression.Call(typeof(Convert),
        "ChangeType", null, valueExpr, Expression.Constant(fieldExpr.Type));
    UnaryExpression valueCast = Expression.Convert(convertExpr, fieldExpr.Type);
    BinaryExpression assignExpr = Expression.Assign(fieldExpr, valueCast);
    return Expression.Lambda<Action<T, string>>(assignExpr, targetExpr, valueExpr);
}
Run Code Online (Sandbox Code Playgroud)