我正在使用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等。
答案 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);
}