您好,登錄后才能下訂單哦!
今天小編給大家分享一下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調用來實現某些需求。本文將介紹兩種方法來實現它。
這種方法的核心思想是,使用反射找到你需要調用的接口實現的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<IService>(m => m.Proccess()); Console.WriteLine("End Proccess"); } }
于是接下來,我們就需要根據lambda表達式找到其對應的接口實現,然后調用即可。
第一步根據lambda表達式獲取MethodInfo和參數。要注意的是,對于屬性的調用我們也需要支持,其實屬性也是一種方法,所以可以一并處理。
private static (MethodInfo method, IReadOnlyList<Expression> args) GetMethodAndArguments(Expression exp) => exp switch { LambdaExpression lambda => GetMethodAndArguments(lambda.Body), UnaryExpression unary => GetMethodAndArguments(unary.Operand), MethodCallExpression methodCall => (methodCall.Method!, methodCall.Arguments), MemberExpression { Member: PropertyInfo prop } => (prop.GetGetMethod(true) ?? throw new MissingMethodException($"No getter in propery {prop.Name}"), Array.Empty<Expression>()), _ => 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 => IfMatch(method, genericArguments, parameters, m)) .ToArray(); var interfaceMethod = interfaceMethods.Length switch { 0 => throw new MissingMethodException($"Can not find method {method.Name} in type {instanceType.Name}"), > 1 => throw new AmbiguousMatchException($"Found more than one method {method.Name} in type {instanceType.Name}"), 1 when interfaceMethods[0].IsAbstract => throw new InvalidOperationException($"The method {interfaceMethods[0].Name} is abstract"), _ => interfaceMethods[0] }; if (method.IsGenericMethod) interfaceMethod = interfaceMethod.MakeGenericMethod(method.GetGenericArguments()); return interfaceMethod; }
第三步,用獲取到的接口方法,構建DynamicMethod
。其中的重點是使用OpCodes.Call
,它的含義是以非虛方式調用一個方法,哪怕該方法是虛方法,也不去查找它的重寫,而是直接調用它自身。
private static DynamicMethod GetDynamicMethod(Type interfaceType, MethodInfo method, IEnumerable<Type> 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">完整的代碼點這里
這個方法和方法1大同小異,區別是,在方法1的第二步,即找到接口方法的MethodInfo
之后,獲取其函數指針,然后利用該指針構造委托。這個方法其實是我最初找到的方法,方法1是其改進。在此就不多做介紹了
方法1雖然可行,但是肉眼可見的性能損失大,即使是用了緩存。于是乎我利用Fody編寫了一個插件InterfaceBaseInvoke.Fody。
其核心思想就是在編譯時找到目標接口方法,然后使用call命令調用它就行了。這樣可以把性能損失降到最低。該插件的使用方法可以參考項目介紹。
方法 | 平均用時 | 內存分配 |
父類的base調用 | 0.0000 ns | - |
方法1(DynamicMethod) | 691.3687 ns | 776 B |
方法2(FunctionPointer) | 1,391.9345 ns | 1,168 B |
方法3(InterfaceBaseInvoke.Fody) | 0.0066 ns | - |
以上就是“C#如何實現接口base調用”這篇文章的所有內容,感謝各位的閱讀!相信大家閱讀完這篇文章都有很大的收獲,小編每天都會為大家更新不同的知識,如果還想學習更多的知識,請關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。