文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

C#如何实现接口base调用

2023-07-02 08:55

关注

今天小编给大家分享一下C#如何实现接口base调用的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。

背景

在三年前发布的C#8.0中有一项重要的改进叫做接口默认实现,从此以后,接口中定义的方法可以包含方法体了,即默认实现。

不过对于接口的默认实现,其实现类或者子接口在重写这个方法的时候不能对其进行base调用,就像子类重写方法是可以进行base.Method()那样。例如:

public interface IService{    void Proccess()    {        Console.WriteLine("Proccessing");    }}public class Service : IService{    public void Proccess()    {        Console.WriteLine("Before Proccess");        base(IService).Proccess(); // 目前不支持,也是本文需要探讨的部分        Console.WriteLine("End Proccess");    }}

当初C#团队将这个特性列为了下一步的计划(点此查看细节),然而三年过去了依然没有被提上日程。这个特性的缺失无疑是一种很大的限制,有时候我们确实需要接口的base调用来实现某些需求。本文将介绍两种方法来实现它。

方法1:使用反射找到接口实现并进行调用

这种方法的核心思想是,使用反射找到你需要调用的接口实现的MethodInfo,然后构建DynamicMethod使用OpCodes.Call去调用它即可。

首先我们定义方法签名用来表示接口方法的base调用。

public static void Base<TInterface>(this TInterface instance, Expression<Action<TInterface>> selector);public static TReturn Base<TInterface, TReturn>(this TInterface instance, Expression<Func<TInterface, TReturn>> selector);

所以上一节的例子就可以改写成:

public class Service : IService{    public void Proccess()    {        Console.WriteLine("Before Proccess");        this.Base&lt;IService&gt;(m =&gt; m.Proccess());        Console.WriteLine("End Proccess");    }}

于是接下来,我们就需要根据lambda表达式找到其对应的接口实现,然后调用即可。

第一步根据lambda表达式获取MethodInfo和参数。要注意的是,对于属性的调用我们也需要支持,其实属性也是一种方法,所以可以一并处理。

private static (MethodInfo method, IReadOnlyList&lt;Expression&gt; args) GetMethodAndArguments(Expression exp) =&gt; exp switch{    LambdaExpression lambda =&gt; GetMethodAndArguments(lambda.Body),    UnaryExpression unary =&gt; GetMethodAndArguments(unary.Operand),    MethodCallExpression methodCall =&gt; (methodCall.Method!, methodCall.Arguments),    MemberExpression { Member: PropertyInfo prop } =&gt; (prop.GetGetMethod(true) ?? throw new MissingMethodException($"No getter in propery {prop.Name}"), Array.Empty&lt;Expression&gt;()),    _ =&gt; throw new InvalidOperationException("The expression refers to neither a method nor a readable property.")};

第二步,利用Type.GetInterfaceMap获取到需要调用的接口实现方法。此处注意的要点是,instanceType.GetInterfaceMap(interfaceType).InterfaceMethods 会返回该接口的所有方法,所以不能仅根据方法名去匹配,因为可能有各种重载、泛型参数、还有new关键字声明的同名方法,所以可以按照方法名+声明类型+方法参数+方法泛型参数唯一确定一个方法(即下面代码块中IfMatch的实现)

internal readonly record struct InterfaceMethodInfo(Type InstanceType, Type InterfaceType, MethodInfo Method);private static MethodInfo GetInterfaceMethod(InterfaceMethodInfo info){    var (instanceType, interfaceType, method) = info;    var parameters = method.GetParameters();    var genericArguments = method.GetGenericArguments();    var interfaceMethods = instanceType        .GetInterfaceMap(interfaceType)        .InterfaceMethods        .Where(m =&gt; IfMatch(method, genericArguments, parameters, m))        .ToArray();    var interfaceMethod = interfaceMethods.Length switch    {        0 =&gt; throw new MissingMethodException($"Can not find method {method.Name} in type {instanceType.Name}"),        &gt; 1 =&gt; throw new AmbiguousMatchException($"Found more than one method {method.Name} in type {instanceType.Name}"),        1 when interfaceMethods[0].IsAbstract =&gt; throw new InvalidOperationException($"The method {interfaceMethods[0].Name} is abstract"),        _ =&gt; interfaceMethods[0]    };    if (method.IsGenericMethod)        interfaceMethod = interfaceMethod.MakeGenericMethod(method.GetGenericArguments());    return interfaceMethod;}

第三步,用获取到的接口方法,构建DynamicMethod。其中的重点是使用OpCodes.Call,它的含义是以非虚方式调用一个方法,哪怕该方法是虚方法,也不去查找它的重写,而是直接调用它自身。

private static DynamicMethod GetDynamicMethod(Type interfaceType, MethodInfo method, IEnumerable&lt;Type&gt; argumentTypes){    var dynamicMethod = new DynamicMethod(        name: "__IL_" + method.GetFullName(),        returnType: method.ReturnType,        parameterTypes: new[] { interfaceType, typeof(object[]) },        owner: typeof(object),        skipVisibility: true);    var il = dynamicMethod.GetILGenerator();    il.Emit(OpCodes.Ldarg_0);    var i = 0;    foreach (var argumentType in argumentTypes)    {        il.Emit(OpCodes.Ldarg_1);        il.Emit(OpCodes.Ldc_I4, i);        il.Emit(OpCodes.Ldelem, typeof(object));        if (argumentType.IsValueType)        {            il.Emit(OpCodes.Unbox_Any, argumentType);        }        ++i;    }    il.Emit(OpCodes.Call, method);    il.Emit(OpCodes.Ret);    return dynamicMethod;}

最后,将DynamicMethod转为强类型的委托就完成了。考虑到性能的优化,可以将最终的委托缓存起来,下次调用就不用再构建一次了。

han12345/c42de446a23aa9a17fb6abf905479f25" rel="external nofollow" target="_blank">完整的代码点这里 

方法2:利用函数指针

这个方法和方法1大同小异,区别是,在方法1的第二步,即找到接口方法的MethodInfo之后,获取其函数指针,然后利用该指针构造委托。这个方法其实是我最初找到的方法,方法1是其改进。在此就不多做介绍了

方法3:利用Fody在编译时对接口方法进行IL的call调用

方法1虽然可行,但是肉眼可见的性能损失大,即使是用了缓存。于是乎我利用Fody编写了一个插件InterfaceBaseInvoke.Fody。

其核心思想就是在编译时找到目标接口方法,然后使用call命令调用它就行了。这样可以把性能损失降到最低。该插件的使用方法可以参考项目介绍。

性能测试

方法平均用时内存分配
父类的base调用0.0000 ns-
方法1(DynamicMethod)691.3687 ns776 B
方法2(FunctionPointer)1,391.9345 ns1,168 B
方法3(InterfaceBaseInvoke.Fody)0.0066 ns-

以上就是“C#如何实现接口base调用”这篇文章的所有内容,感谢各位的阅读!相信大家阅读完这篇文章都有很大的收获,小编每天都会为大家更新不同的知识,如果还想学习更多的知识,请关注编程网行业资讯频道。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     807人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     351人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     314人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     433人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     221人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯