欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解C#通過反射獲取對象的幾種方式比較

 更新時間:2022年06月14日 09:19:58   作者:HappyGirl快樂女孩  
本文主要介紹了C#通過反射獲取對象的幾種方式比較,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧

在本文中,對比了常見的幾種反射的方法,介紹了它們分別應該如何使用,每種的簡易度和靈活度,然后做了基準測試,一起看看這之間的性能差距。

按照使用的簡易度和靈活度,做了下邊的排序,可能還有一些其他的反射方式,比如 Source Generators,本文中只針對以下幾種進行測試。

  • 直接調(diào)用 ConstructorInfo 對象的Invoke()方法
  • 使用 Activator.CreateInstance()
  • 使用 Microsoft.Extensions.DependencyInjection
  • 黑科技 Natasha•使用表達式 Expression
  • 使用 Reflection.Emit 創(chuàng)建動態(tài)方法

使用標準反射的 Invoke 方法

Type typeToCreate = typeof(Employee);
ConstructorInfo ctor = typeToCreate.GetConstructor(System.Type.EmptyTypes);
Employee employee = ctor.Invoke(null) as Employee;

第一步是通過 typeof() 獲取對象的類型,你也可以通過 GetType 的方式,然后調(diào)用 GetConstructor 方法,傳入 System.Type.EmptyTypes 參數(shù),實際上它是一個空數(shù)組 (new Type[0]), 返回 ConstructorInfo對象, 然后調(diào)用 Invoke 方法,會返回一個 Employee 對象。

這是使用反射的最簡單和最靈活的方法之一,因為可以使用類似的方法來調(diào)用對象的方法、接口和屬性等,但是這個也是最慢的反射方法之一。

使用 Activator.CreateInstance

如果你需要創(chuàng)建對象的話,在.NET Framework 和 .NET Core 中正好有一個專門為此設計的靜態(tài)類,System.Activator, 使用方法非常的簡單,還可以使用泛型,而且你還可以傳入其他的參數(shù)。

Employee employee = Activator.CreateInstance<Employee>();

使用 Microsoft.Extensions.DependencyInjection

接下來就是在.NET Core 中很熟悉的 IOC 容器,Microsoft.Extensions.DependencyInjection,把類型注冊到容器中后,然后使用 IServiceProvider 來獲取對象,這里使用了 Transient 的生命周期,保證每次都會創(chuàng)建一個新的對象

IServiceCollection services = new ServiceCollection();
 
services.AddTransient<Employee>();
 
IServiceProvider provider = services.BuildServiceProvider();
 
Employee employee = provider.GetService<Employee>();

Natasha

Natasha 是基于 Roslyn 開發(fā)的動態(tài)程序集構建庫,直觀和流暢的 Fluent API 設計,通過 roslyn 的強大賦能, 可以在程序運行時創(chuàng)建代碼,包括 程序集、類、結構體、枚舉、接口、方法等, 用來增加新的功能和模塊,這里用 NInstance 來創(chuàng)建對象。

// Natasha 初始化
NatashaInitializer.Initialize();
 
Employee employee = Natasha.CSharp.NInstance.Creator<Employee>().Invoke();

使用表達式 Expression

表達式 Expression 其實也已經(jīng)存在很長時間了,在 System.Linq.Expressions 命名空間下, 并且是各種其他功能 (LINQ) 和庫(EF Core) 不可或缺的一部分,在許多方面,它類似于反射,因為它們允許在運行時操作代碼。

NewExpression constructorExpression = Expression.New(typeof(Employee));
Expression<Func<Employee>> lambdaExpression = Expression.Lambda<Func<Employee>>(constructorExpression);
Func<Employee> func = lambdaExpression.Compile();
Employee employee = func();

 表達式提供了一種用于聲明式代碼的高級語言,前兩行創(chuàng)建了的表達式, 等價于 () => new Employee(),然后調(diào)用 Compile 方法得到一個 Func<> 的委托,最后調(diào)用這個 Func 返回一個Employee對象

使用 Emit

Emit 主要在 System.Reflection.Emit 命名空間下,這些方法允許在程序中直接創(chuàng)建 IL (中間代碼) 代碼,IL 代碼是指編譯器在編譯程序時輸出的 "偽匯編代碼", 也就是編譯后的dll,當程序運行的時候,.NET CLR 中的 JIT編譯器 將這些 IL 指令轉(zhuǎn)換為真正的匯編代碼。

接下來,需要在運行時創(chuàng)建一個新的方法,很簡單,沒有參數(shù),只是創(chuàng)建一個Employee對象然后直接返回

Employee DynamicMethod()
{
    return new Employee();
}

這里主要使用到了 System.Reflection.Emit.DynamicMethod 動態(tài)創(chuàng)建方法 

 DynamicMethod dynamic = new("DynamicMethod", typeof(Employee), null, typeof(ReflectionBenchmarks).Module, false);

創(chuàng)建了一個 DynamicMethod 對象,然后指定了方法名,返回值,方法的參數(shù)和所在的模塊,最后一個參數(shù) false 表示不跳過 JIT 可見性檢查。

現(xiàn)在有了方法簽名,但是還沒有方法體,還需要填充方法體,這里需要C#代碼轉(zhuǎn)換成 IL代碼,實際上它是這樣的

IL_0000: newobj instance void Employee::.ctor()

IL_0005: ret

然后使用 ILGenerator 來操作IL代碼, 然后創(chuàng)建一個 Func<> 的委托, 最后執(zhí)行該委托返回一個 Employee 對象

ConstructorInfor ctor = typeToCreate.GetConstructor(System.Type.EmptyTypes);
 
ILGenerator il = createHeadersMethod.GetILGenerator();
il.Emit(OpCodes.Newobj, Ctor);
il.Emit(OpCodes.Ret);
 
Func<Employee> emitActivator = dynamic.CreateDelegate(typeof(Func<Employee>)) as Func<Employee>;
Employee employee = emitActivator();

對比測試

using BenchmarkDotNet.Attributes;
using Microsoft.Extensions.DependencyInjection;
using System;
using System.Linq.Expressions;
using System.Reflection;
using System.Reflection.Emit;
 
namespace ReflectionBenchConsoleApp
{
    public class Employee { }
 
    public class ReflectionBenchmarks
    { 
        private readonly ConstructorInfo _ctor;
        private readonly IServiceProvider _provider;
        private readonly Func<Employee> _expressionActivator;
        private readonly Func<Employee> _emitActivator;
        private readonly Func<Employee> _natashaActivator;
 
 
        public ReflectionBenchmarks()
        { 
            _ctor = typeof(Employee).GetConstructor(Type.EmptyTypes); 
 
            _provider = new ServiceCollection().AddTransient<Employee>().BuildServiceProvider(); 
 
            NatashaInitializer.Initialize();
            _natashaActivator = Natasha.CSharp.NInstance.Creator<Employee>();
 
 
            _expressionActivator = Expression.Lambda<Func<Employee>>(Expression.New(typeof(Employee))).Compile(); 
 
            DynamicMethod dynamic = new("DynamicMethod", typeof(Employee), null, typeof(ReflectionBenchmarks).Module, false);  
            ILGenerator il = dynamic.GetILGenerator();
            il.Emit(OpCodes.Newobj, typeof(Employee).GetConstructor(System.Type.EmptyTypes));
            il.Emit(OpCodes.Ret); 
            _emitActivator = dynamic.CreateDelegate(typeof(Func<Employee>)) as Func<Employee>;  
 
        }  
 
        [Benchmark(Baseline = true)]
        public Employee UseNew() => new Employee(); 
 
        [Benchmark]
        public Employee UseReflection() => _ctor.Invoke(null) as Employee;
 
        [Benchmark]
        public Employee UseActivator() => Activator.CreateInstance<Employee>();  
 
        [Benchmark]
        public Employee UseDependencyInjection() => _provider.GetRequiredService<Employee>();
 
 
        [Benchmark]
        public Employee UseNatasha() => _natashaActivator();
 
 
        [Benchmark]
        public Employee UseExpression() => _expressionActivator(); 
 
 
        [Benchmark]
        public Employee UseEmit() => _emitActivator(); 
 
 
    }  
}

接下來,還修改 Program.cs,注意這里需要在 Release 模式下運行測試

using BenchmarkDotNet.Running; 
 
namespace ReflectionBenchConsoleApp
{
    public class Program
    {
        public static void Main(string[] args)
        { 
            var sumary = BenchmarkRunner.Run<ReflectionBenchmarks>();
        }
    } 
 
}

測試結果

 環(huán)境是 .NET 6 preview5, 使用標準反射的 Invoke() 方法雖然簡單,但它是最慢的一種,使用 Activator.CreateInstance() 和 Microsoft.Extensions.DependencyInjection() 的時間差不多,時間是直接 new 創(chuàng)建的16倍,使用表達式 Expression 表現(xiàn)最優(yōu)秀,Natasha 真是黑科技,比用Emit 還快了一點,使用Emit 是直接 new 創(chuàng)建的時間的1.8倍。你應該發(fā)現(xiàn)了各種方式之間的差距,但是需要注意的是這里是 ns 納秒,一納秒是一秒的十億分之一。

總結

這里簡單對比了幾種創(chuàng)建對象的方法,測試的結果也可能不是特別準確,有興趣的還可以在 .net framework 上面進行測試,希望對您有用!

相關鏈接 

https://andrewlock.net/benchmarking-4-reflection-methods-for-calling-a-constructor-in-dotnet/

https://github.com/dotnetcore/Natasha

到此這篇關于詳解C#通過反射獲取對象的幾種方式比較的文章就介紹到這了,更多相關C# 反射獲取對象內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • C#之Windows自帶打印功能的實現(xiàn)

    C#之Windows自帶打印功能的實現(xiàn)

    這篇文章主要介紹了C#之Windows自帶打印功能的實現(xiàn)方式,具有很好的價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-06-06
  • 淺談C#手機號換成111XXXX1111 這種顯示的解決思路

    淺談C#手機號換成111XXXX1111 這種顯示的解決思路

    下面小編就為大家?guī)硪黄獪\談C#手機號換成111XXXX1111 這種顯示的解決思路。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-11-11
  • C# 常用協(xié)議實現(xiàn)模版及FixedSizeReceiveFilter示例(SuperSocket入門)

    C# 常用協(xié)議實現(xiàn)模版及FixedSizeReceiveFilter示例(SuperSocket入門)

    本文主要介紹了常用協(xié)議實現(xiàn)模版及FixedSizeReceiveFilter示例。具有很好的參考價值,下面跟著小編一起來看下吧
    2017-01-01
  • C#公眾號開發(fā)之給用戶發(fā)紅包

    C#公眾號開發(fā)之給用戶發(fā)紅包

    這篇文章主要為大家詳細介紹了C#公眾號開發(fā)之給用戶發(fā)紅包,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-08-08
  • C#實現(xiàn)獲取機器碼的示例詳解

    C#實現(xiàn)獲取機器碼的示例詳解

    這篇文章主要為大家詳細介紹了如何利用C#實現(xiàn)獲取機器碼的功能,文中的示例代碼講解詳細,對我們學習C#有一定的幫助,感興趣的小伙伴可以跟隨小編一起了解一下
    2022-12-12
  • C#使用System.Net.Mail類實現(xiàn)郵件發(fā)送

    C#使用System.Net.Mail類實現(xiàn)郵件發(fā)送

    這篇文章介紹了C#使用System.Net.Mail類實現(xiàn)郵件發(fā)送的方法,文中通過示例代碼介紹的非常詳細。對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-07-07
  • C#十五子游戲編寫代碼

    C#十五子游戲編寫代碼

    這篇文章主要為大家詳細介紹了C#十五子游戲的編寫代碼,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-05-05
  • C#實現(xiàn)提高xml讀寫速度的方法

    C#實現(xiàn)提高xml讀寫速度的方法

    這篇文章主要介紹了C#實現(xiàn)提高xml讀寫速度的方法,并且針對各類文件的讀寫做了較為細致的分析,非常實用,需要的朋友可以參考下
    2014-11-11
  • c#閉包使用方法示例

    c#閉包使用方法示例

    這篇文章主要介紹了如何使用C#的閉包功用,例子很簡單,大家參考使用吧
    2013-11-11
  • C#中實現(xiàn)查找mysql的安裝路徑

    C#中實現(xiàn)查找mysql的安裝路徑

    這篇文章主要介紹了C#中實現(xiàn)查找mysql的安裝路徑,本文講解使用SQL語句查詢出mysql的安裝路徑,方便在備份時使用,需要的朋友可以參考下
    2015-06-06

最新評論