Я использую Mono.Cecil
(версия 0.6.9.0) для псевдонима метода, реализующего метод интерфейса. Чтобы сделать это, я должен добавить Override
к целевому методу, который указывает на метод интерфейса (так же, как это возможно в VB.NET), например:
using System;
using System.Reflection;
using Mono.Cecil;
class Program {
static void Main(string[] args) {
var asm = AssemblyFactory.GetAssembly(Assembly.GetExecutingAssembly().Location);
var source = asm.MainModule.Types["A"];
var sourceMethod = source.Methods[0];
var sourceRef = new MethodReference(
sourceMethod.Name,
sourceMethod.DeclaringType,
sourceMethod.ReturnType.ReturnType,
sourceMethod.HasThis,
sourceMethod.ExplicitThis,
sourceMethod.CallingConvention);
var target = asm.MainModule.Types["C"];
var targetMethod = target.Methods[0];
targetMethod.Name = "AliasedMethod";
targetMethod.Overrides.Add(sourceRef);
AssemblyAssert.Verify(asm); // this will just run PEVerify on the changed assembly
}
}
interface A {
void Method();
}
class C : A {
public void Method() { }
}
Я получаю ошибку PEVerify.exe
, указывающую, что мой класс больше не реализует метод интерфейса. Кажется, что в измененной сборке есть несоответствие токена между ссылкой переопределения и методом в интерфейсе:
[MD]: Error: Class implements interface but not method (class:0x02000004; interface:0x02000002; method:0x06000001). [token:0x09000001]
Я знаю, что если я использую MethodDefinition
напрямую, это сработает:
targetMethod.Overrides.Add(sourceMethod);
Но мне действительно нужно использовать MethodReference
, потому что у меня могут быть общие параметры и аргументы в задействованных типах, а простой MethodDefinition
не подойдет.
Обновление: в соответствии с рекомендациями Jb Evain, я мигрировал на версию 0.9 .3.0. Но та же ошибка все еще происходит. Вот перенесенный код:
var module = ModuleDefinition.ReadModule(Assembly.GetExecutingAssembly().Location);
var source = module.GetType("A");
var sourceMethod = source.Methods[0];
var sourceRef = new MethodReference(
sourceMethod.Name,
sourceMethod.ReturnType) {
DeclaringType = sourceMethod.DeclaringType,
HasThis = sourceMethod.HasThis,
ExplicitThis = sourceMethod.ExplicitThis,
CallingConvention = sourceMethod.CallingConvention
};
var target = module.GetType("C");
var targetMethod = target.Methods[0];
targetMethod.Name = "AliasedMethod";
targetMethod.Overrides.Add(sourceRef);