如何使用Mono.Cecil创建覆盖方法?

时间:2011-11-12 02:12:47

标签: mono.cecil

我正在使用Mono.Cecil生成一个程序集,该程序集包含一个派生类,该类覆盖导入的基类中的特定方法。覆盖方法是一个“隐式”覆盖。问题是我无法弄清楚如何将其指定为覆盖。

我正在使用以下代码创建覆盖方法。

    void CreateMethodOverride(TypeDefinition targetType,
        TypeDefinition baseClass, string methodName, MethodInfo methodInfo)
    {
        // locate the matching base class method, which may
        // reside in a different module
        MethodDefinition baseMethod = baseClass
            .Methods.First(method => method.Name.Equals(methodName));

        MethodDefinition newMethod = targetType.Copy(methodInfo);
        newMethod.Name = baseMethod.Name;
        newMethod.Attributes = baseMethod.Attributes;
        newMethod.ImplAttributes = baseMethod.ImplAttributes;
        newMethod.SemanticsAttributes = baseMethod.SemanticsAttributes;
        targetType.Methods.Add(newMethod);
    }

我理解隐式覆盖必须与继承方法具有相同的签名。使用上面的代码,当我在Reflector中查看结果方法时,基类和派生类方法具有完全相同的签名,即“public virtual void f(int param)”。

我尝试删除显式的“虚拟”属性,但派生的方法最终为“public void f(int param)”。

如何让派生方法拥有正确的“public override void f(int param)”签名?

注意:我有一个扩展方法(“TypeDefinition.Copy”),它克隆MethodInfo并通过导入所有引用的类型返回MethodDefinition等。

2 个答案:

答案 0 :(得分:10)

在您的基类中,假设您生成以下方法:

public virtual void f(int);

您必须确保将IsVirtual标志设置为true。您还必须确保它具有标记IsNewSlot = true,以确保它在virtual method table中有新的插槽。

现在,对于重写的方法,您希望生成:

public override void f(int);

要做到这一点,你还需要让方法为IsVirtual,但也要告诉它它不是一个新的虚方法,而是隐含地覆盖另一个,所以你必须使它成为.IsReuseSlot = true。 1}}。

由于您使用的是隐式覆盖,因此您还必须确保这两种方法都是.IsHideBySig = true

有了这一切,你应该有一个合适的重写方法。

答案 1 :(得分:7)

为了其他读者的利益,以下是按照JB的答案获得的最终结果:

void CreateMethodOverride(TypeDefinition targetType,
    TypeDefinition baseClass, string methodName, MethodInfo methodInfo)
{
    MethodDefinition baseMethod = baseClass
        .Methods.First(method => method.Name.Equals(methodName));

    MethodDefinition newMethod = targetType.Copy(methodInfo);
    newMethod.Name = baseMethod.Name;

    // Remove the 'NewSlot' attribute
    newMethod.Attributes = baseMethod.Attributes & ~MethodAttributes.NewSlot;

    // Add the 'ReuseSlot' attribute
    newMethod.Attributes |= MethodAttributes.ReuseSlot;

    newMethod.ImplAttributes = baseMethod.ImplAttributes;
    newMethod.SemanticsAttributes = baseMethod.SemanticsAttributes;
    targetType.Methods.Add(newMethod);
}