詳解C#實(shí)例化對(duì)象的三種方式及性能對(duì)比
前言
做項(xiàng)目過程中有個(gè)需求要實(shí)例化兩萬個(gè)對(duì)象并添加到List
中,這個(gè)過程大概需要1min才能加載完(傳參較多),于是開啟了代碼優(yōu)化之旅,再此記錄。
首先想到的是可能實(shí)例化比較耗時(shí),于是開始對(duì)每種實(shí)例化方式進(jìn)行測(cè)試,過程如下
實(shí)例化方式
1、用 New 關(guān)鍵字實(shí)例化一個(gè)類
2、用 Activator 實(shí)例化一個(gè)類
3、用 Assembly 實(shí)例化一個(gè)類
代碼實(shí)現(xiàn)
測(cè)試環(huán)境:
vs2019 .NET Framework 4.7
Intel Core i7-10510U CPU
首先定義一個(gè)類Person
public class Person { public Person() { } public Person(string name) { Name = name; } public string Name { get; set; } }
我們先在無參的構(gòu)造函數(shù)中實(shí)例化,每種方式進(jìn)行十次,每次實(shí)例化十萬次,代碼如下
static void Main(string[] args) { Console.WriteLine("實(shí)例化對(duì)象的耗時(shí)比較(單位:毫秒)"); Console.Write(" "); for (int i = 1; i <= 10; i++) Console.Write("{0:G}", i.ToString().PadLeft(5)); Console.Write("\n"); Console.Write("InstanceByNew".PadRight(20)); for (int i = 1; i <= 10; i++) { Person person = null; Stopwatch watch = new Stopwatch(); watch.Start(); for (int j = 0; j < 100000; j++) person = new Person(); watch.Stop(); Console.Write(watch.ElapsedMilliseconds.ToString().PadLeft(5)); } Console.Write("\n"); Console.Write("InstanceByActivator".PadRight(20)); for (int i = 1; i <= 10; i++) { Type type = Type.GetType("ConsoleApp1.Person"); Person person = null; Stopwatch watch = new Stopwatch(); watch.Start(); for (int j = 0; j < 100000; j++) { object obj = Activator.CreateInstance(type); person = obj as Person; } watch.Stop(); Console.Write(watch.ElapsedMilliseconds.ToString().PadLeft(5)); } Console.Write("\n"); Console.Write("InstanceByAssembly".PadRight(20)); for (int i = 1; i <= 10; i++) { Assembly assembly = Assembly.Load("InstancePerformance"); Person person = null; Stopwatch watch = new Stopwatch(); watch.Start(); for (int j = 0; j < 100000; j++) { object obj = assembly.CreateInstance("ConsoleApp1.Person"); person = obj as Person; } watch.Stop(); Console.Write(watch.ElapsedMilliseconds.ToString().PadLeft(5)); } Console.Write("\n"); Console.ReadKey(); }
執(zhí)行結(jié)果如下:
然后來看下有參構(gòu)造函數(shù)中實(shí)例化的代碼和結(jié)果
static void Main(string[] args) { Console.WriteLine("實(shí)例化對(duì)象的耗時(shí)比較(單位:毫秒)"); Console.Write(" "); for (int i = 1; i <= 10; i++) Console.Write("{0:G}", i.ToString().PadLeft(5)); Console.Write("\n"); Console.Write("InstanceByNew".PadRight(20)); for (int i = 1; i <= 10; i++) { Person person = null; Stopwatch watch = new Stopwatch(); watch.Start(); for (int j = 0; j < 100000; j++) person = new Person("Test"+j); watch.Stop(); Console.Write(watch.ElapsedMilliseconds.ToString().PadLeft(5)); } Console.Write("\n"); Console.Write("InstanceByActivator".PadRight(20)); for (int i = 1; i <= 10; i++) { Type type = Type.GetType("ConsoleApp1.Person"); Person person = null; Stopwatch watch = new Stopwatch(); watch.Start(); for (int j = 0; j < 100000; j++) { object obj = Activator.CreateInstance(type,new object[]{"Test"+j}); person = obj as Person; } watch.Stop(); Console.Write(watch.ElapsedMilliseconds.ToString().PadLeft(5)); } Console.Write("\n"); Console.Write("InstanceByAssembly".PadRight(20)); for (int i = 1; i <= 10; i++) { Assembly assembly = Assembly.Load("InstancePerformance"); Person person = null; Stopwatch watch = new Stopwatch(); watch.Start(); for (int j = 0; j < 100000; j++) { object obj = assembly.CreateInstance("ConsoleApp1.Person", true, System.Reflection.BindingFlags.Default, null, new []{"Test"+j}, null, null); person = obj as Person; } watch.Stop(); Console.Write(watch.ElapsedMilliseconds.ToString().PadLeft(5)); } Console.Write("\n"); Console.ReadKey(); }
執(zhí)行結(jié)果如下:
結(jié)論
從上面的執(zhí)行結(jié)果可以看出這三種方式的性能排序?yàn)?/p>
New > Activator > Assembly
但使用哪種方法還要視情況而定
后續(xù)
以上可以看出New性能最高,而我就是使用的New,問題還沒解決,把問題指向判斷List是否存在某元素上,于是開始測(cè)試List中的Contains, Exists, Any,Where。詳情請(qǐng)點(diǎn)擊.
到此這篇關(guān)于C#實(shí)例化對(duì)象的三種方式及性能對(duì)比的文章就介紹到這了,更多相關(guān)C#實(shí)例化對(duì)象的三種方式及性能對(duì)比內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C#中的delegate委托類型基本學(xué)習(xí)教程
這篇文章主要介紹了C#中的delegate委托類型基本學(xué)習(xí)教程,委托是C#語言所具有的一個(gè)重要特性,需要的朋友可以參考下2016-01-01DevExpress之ChartControl實(shí)現(xiàn)柱狀圖演示實(shí)例
這篇文章主要介紹了DevExpress中ChartControl實(shí)現(xiàn)柱狀圖演示方法,實(shí)例展示了相關(guān)繪圖函數(shù)的具體用法,具有一定的實(shí)用價(jià)值,需要的朋友可以參考下2014-10-10winfrom 打印表格 字符串的封裝實(shí)現(xiàn)代碼 附源碼下載
以前寫打印都是根據(jù)打印機(jī)的型號(hào),找開發(fā)類庫。然后在此基礎(chǔ)上開發(fā)。寫的多了自然就想到了封裝。這是還是想到了微軟,微軟封裝了PrintDocument的打印類。但這只是在低層對(duì)串口的封裝2013-02-02C# 使用PictureBox實(shí)現(xiàn)圖片按鈕控件的示例步驟
這篇文章主要介紹了C# 使用PictureBox實(shí)現(xiàn)圖片按鈕控件的示例步驟,幫助大家更好的理解和使用c#,感興趣的朋友可以了解下2021-02-02C#實(shí)現(xiàn)簡(jiǎn)單的計(jì)算器功能(窗體)
這篇文章主要為大家詳細(xì)介紹了C#實(shí)現(xiàn)簡(jiǎn)單的計(jì)算器功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01