8

我正在尝试创建一个通用操作代表

  delegate void ActionPredicate<in T1, in T2>(T1 t1, T2 t2);

public static ActionPredicate<T,string> GetSetterAction<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);
        var result = Expression.Lambda<ActionPredicate<T, string>>(assignExpr, targetExpr, valueExpr);
        return result.Compile();
    }

这是我的来电者

 ActionPredicate<busBase, string> act = DelegateGenerator.GetSetterAction<busPerson>("FirstName");

这是业务对象

 public abstract class busBase 
{

}
public class busPerson : busBase
{
    public string FirstName { get; set; }
    public string LastName { get; set; }
    public int Age { get; set; }

    public string GetFullName()
    {
        return string.Format("{0} {1}", FirstName, LastName);
    }
}

这是我在编译过程中得到的错误

Cannot implicitly convert type 'BusinessObjects.ActionPredicate<BusinessObjects.busPerson,string>' to 'BusinessObjects.ActionPredicate<BusinessObjects.busBase,string>'. An explicit conversion exists (are you missing a cast?)    

我的 GetSetterAction 正在返回 ActionPerdicate ,因为这里 T 是 busPerson ,我试图将它存储在 ActionPredicate 中,记住逆变。但它失败了。我不知道如何进一步进行。请帮忙..!

4

1 回答 1

8

通用逆变不允许您将委托分配给委托D<TDerived>D<TBase>原因如下所示(Action<T1>在此处使用):

Action<string> m1 = MyMethod; //some method to call
Action<object> m2 = m1; //compiler error - but pretend it's not.
object obj = new object();

m2(obj);  //runtime error - not type safe

如您所见,如果我们被允许执行此分配,那么我们将破坏类型安全,因为我们将能够尝试通过传递 and和 notm1的实例来调用委托。然而,换一种方式,即将委托引用复制到其参数类型比源更派生的类型是可以的。 MSDN 有一个更完整的通用 co/contra variance 示例objectstring

因此,您将需要更改 to 的声明,act或者ActionPredicate<busPerson, string> act更有可能考虑编写GetSetterAction始终返回的方法ActionPredicate<busBase, string>。如果你这样做,你还应该添加类型约束

where T1 : busBase

对于该方法,您还需要更改表达式的构建方式,将前两行替换如下:

ParameterExpression targetExpr = Expression.Parameter(typeof(busBase), "Target");
//generate a strongly-typed downcast to the derived type from busBase and
//use that as the type on which the property is to be written
MemberExpression fieldExpr = Expression.Property(
  Expression.Convert(targetExpr, typeof(T1)), fieldName);

添加通用约束是一个很好的接触,以确保这种向下转换将始终对任何T1.

稍有不同 -Action<T1, T2>代表出了什么问题?它似乎和你做的一模一样?:)

于 2012-04-11T08:00:55.443 回答