亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

.NET獲取枚舉DescriptionAttribute描述信息性能改進的多種方法

2019-11-14 13:34:20
字體:
來源:轉載
供稿:網友

一. DescriptionAttribute的普通使用方式

1.1 使用示例

  DescriptionAttribute特性可以用到很多地方,比較常見的就是枚舉,通過獲取枚舉上定義的描述信息在UI上顯示,一個簡單的枚舉定義:

public enum EnumGender        {            None,            [System.ComponentModel.Description("")]            Male,            [System.ComponentModel.Description("")]            Female,            Other,        }

  本文不討論DescriptionAttribute的其他應用場景,也不關注多語言的實現,只單純的研究下獲取枚舉描述信息的方法。

  一般比較常見的獲取枚舉描述信息的方法如下,可以在園子里搜索類似的代碼非常多。

public static string GetDescriptionOriginal(this Enum @this)        {            var name = @this.ToString();            var field = @this.GetType().GetField(name);            if (field == null) return name;            var att = System.Attribute.GetCustomAttribute(field, typeof(DescriptionAttribute), false);            return att == null ? field.Name : ((DescriptionAttribute)att).Description;        }

  簡單測試下:

Console.WriteLine(EnumGender.Female.GetDescriptionOriginal());Console.WriteLine(EnumGender.Male.GetDescriptionOriginal());Console.WriteLine(EnumGender.Other.GetDescriptionOriginal());
//輸出結果:



Other

1.2 上面的實現代碼的問題

  首先要理解特性是什么?

特性

    Attribute特性就是關聯了一個目標對象的一段配置信息,存儲在dll內的元數據。它本身沒什么意義,可以通過反射來獲取配置的特性信息。

  因此主要問題其實就是反射造成的嚴重性能問題:

  • 1.每次調用都會使用反射,效率慢!
  • 2.每次調用反射都會生成新的DescriptionAttribute對象,哪怕是同一個枚舉值。造成內存、GC的極大浪費!
  • 3.好像不支持位域組合對象!
  • 4.這個地方的方法參數是Enum,Enum是枚舉的基類,他是一個引用類型,而枚舉是值類型,該方法會造成裝箱,不過這個問題好像是不可避免的。

  性能到底有多差呢?代碼來實測一下:

        [Test]        public void GetDescriptionOriginal_Test()        {            var enums = this.GetTestEnums();            Console.WriteLine(enums.Count);            TestHelper.InvokeAndWriteAll(() =>            {                System.Threading.Tasks.Parallel.For(0, 1000000, (i, obj) =>                {                    foreach (var item in enums)                    {                        var a = item.GetDescriptionOriginal();                    }                });            });        }//輸出結果:80TimeSpan:79,881.0000ms //共消耗了將近80秒MemoryUsed:-1,652.7970KBCollectionCount(0):7,990.00  //0代GC回收了7千多次,因為創建了大量的DescriptionAttribute對象

  其中this.GetTestEnums();方法使用獲取一個枚舉值集合,用于測試的,集合大小80,執行100w次,相當于執行了8000w次GetDescriptionOriginal方法。

  TestHelper.InvokeAndWriteAll方法是用來計算執行前后的時間、內存消耗、0代GC回收次數的,文末附錄中給出了代碼,由于內存回收的原因,內存消耗計算其實不準確的,不過可以參考第三個指標0代GC回收次數。

二. 改進的DescriptionAttribute方法

  知道了問題原因,解決就好辦了,基本思路就是把獲取到的文本值緩存起來,一個枚舉值只反射一次,這樣性能問題就解決了。

2.1 使用字典緩存+鎖

  因為使用靜態變量字典來緩存值,就涉及到線程安全,需要使用鎖(做了雙重檢測),具體方法:

PRivate static Dictionary<Enum, string> _LockDictionary = new Dictionary<Enum, string>();        public static string GetDescriptionByDictionaryWithLocak(this Enum @this)        {            if (_LockDictionary.ContainsKey(@this)) return _LockDictionary[@this];            Monitor.Enter(_obj);            if (!_LockDictionary.ContainsKey(@this))            {                var value = @this.GetDescriptionOriginal();                _LockDictionary.Add(@this, value);            }            Monitor.Exit(_obj);            return _LockDictionary[@this];        }

  來測試一下,測試數據、次數和1.2的GetDescriptionOriginal_Test相同,效率有很大的提升,只有一次內存回收。

[Test]        public void GetDescriptionByDictionaryWithLocak_Test()        {            var enums = this.GetTestEnums();            Console.WriteLine(enums.Count);            TestHelper.InvokeAndWriteAll(() =>            {                System.Threading.Tasks.Parallel.For(0, 1000000, (i, obj) =>                {                    foreach (var item in enums)                    {                        var a = item.GetDescriptionByDictionaryWithLocak();                    }                });            });        }//測試結果:80TimeSpan:1,860.0000msMemoryUsed:159.2422KBCollectionCount(0):1.00

2.2 使用字典緩存+異常(不走尋常路的方式)

  還是先看看實現方法吧!

private static Dictionary<Enum, string> _ExceptionDictionary = new Dictionary<Enum, string>();        public static string GetDescriptionByDictionaryWithException(this Enum @this)        {            try            {                return _ExceptionDictionary[@this];            }            catch (KeyNotFoundException)            {                Monitor.Enter(_obj);                if (!_ExceptionDictionary.ContainsKey(@this))                {                    var value = @this.GetDescriptionOriginal();                    _ExceptionDictionary.Add(@this, value);                }                Monitor.Exit(_obj);                return _ExceptionDictionary[@this];            }        }

  假設我們的使用場景是這樣的:項目定義的枚舉并不多,但是用其描述值很頻繁,比如定義了一個用戶性別枚舉,用的地方很多,使用頻率很高。

  上面GetDescriptionByDictionaryWithLocak的方法中,第一句代碼“if (_LockDictionary.ContainsKey(@this)) ”就是驗證是否包含枚舉值。在2.1的測試中執行了8000w次,其中只有80次(總共只有80個枚舉值用于測試)需要這句代碼“if (_LockDictionary.ContainsKey(@this)) ”,其余的直接取值就可了?;谶@樣的考慮,就有了上面的方法GetDescriptionByDictionaryWithException。

  來測試一下,看看效果吧!

[Test]        public void GetDescriptionByDictionaryWithException_Test()        {            var enums = this.GetTestEnums();            Console.WriteLine(enums.Count);            TestHelper.InvokeAndWriteAll(() =>            {                System.Threading.Tasks.Parallel.For(0, 1000000, (i, obj) =>                {                    foreach (var item in enums)                    {                        var a = item.GetDescriptionByDictionaryWithException();                    }                });            });        }//測試結果:80TimeSpan:1,208.0000msMemoryUsed:230.9453KBCollectionCount(0):1.00

  測試結果來看,基本上差不多,在時間上略微快樂一點點,1,208.0000ms:1,860.0000ms,執行8000w次快600毫秒,好像差別也不大啊,這是為什么呢?

  這個其實就是Dictionary的問題了,Dictionary內部使用散列算法計算存儲地址,其查找的時間復雜度為o(1),他的查找效果是非??斓模痉椒ㄖ欣昧水惓L幚?,異常捕獲本身是有一定性能影響的。

2.3 推薦簡單方案:ConcurrentDictionary

  ConcurrentDictionary是一個線程安全的字典類,代碼:

private static ConcurrentDictionary<Enum, string> _ConcurrentDictionary = new ConcurrentDictionary<Enum, string>();        public static string GetDescriptionByConcurrentDictionary(this Enum @this)        {            return _ConcurrentDictionary.GetOrAdd(@this, (key) =>            {                var type = key.GetType();                var field = type.GetField(key.ToString());                return field == null ? key.ToString() : GetDescription(field);            });        }

  測試代碼及測試結果:

[Test]        public void GetDescriptionByConcurrentDictionary_Test()        {            var enums = this.GetTestEnums();            Console.WriteLine(enums.Count);            TestHelper.InvokeAndWriteAll(() =>            {                System.Threading.Tasks.Parallel.For(0, 1000000, (i, obj) =>                {                    foreach (var item in enums)                    {                        var a = item.GetDescriptionByConcurrentDictionary();                    }                });            });        }//測試結果:80TimeSpan:1,303.0000msMemoryUsed:198.0859KBCollectionCount(0):1.00

2.4 正式的代碼

  綜上所述,解決了性能問題、位域枚舉問題的正式的代碼:

/// <summary>        /// 獲取枚舉的描述信息(Descripion)。        /// 支持位域,如果是位域組合值,多個按分隔符組合。        /// </summary>        public static string GetDescription(this Enum @this)        {            return _ConcurrentDictionary.GetOrAdd(@this, (key) =>            {                var type = key.GetType();                var field = type.GetField(key.ToString());                //如果field為null則應該是組合位域值,                return field == null ? key.GetDescriptions() : GetDescription(field);            });        }        /// <summary>        /// 獲取位域枚舉的描述,多個按分隔符組合        /// </summary>        public static string GetDescriptions(this Enum @this, string separator = ",")        {            var names = @this.ToString().Split(',');            string[] res = new string[names.Length];            var type = @this.GetType();            for (int i = 0; i < names.Length; i++)            {                var field = type.GetField(names[i].Trim());                if (field == null) continue;                res[i] = GetDescription(field);            }            return string.Join(separator, res);        }        private static string GetDescription(FieldInfo field)        {            var att = System.Attribute.GetCustomAttribute(field, typeof(DescriptionAttribute), false);            return att == null ? field.Name : ((DescriptionAttribute)att).Description;        }

 版權所有,文章來源:http://www.49028c.com/anding

個人能力有限,本文內容僅供學習、探討,歡迎指正、交流。

附錄:

1.EnumExtension.cs代碼: 

   public static class EnumExtension    {        /// <summary>        /// 獲取枚舉的描述信息(Descripion)。        /// 支持位域,如果是位域組合值,多個按分隔符組合。        /// </summary>        public static string GetDescription(this Enum @this)        {            return _ConcurrentDictionary.GetOrAdd(@this, (key) =>            {                var type = key.GetType();                var field = type.GetField(key.ToString());                //如果field為null則應該是組合位域值,                return field == null ? key.GetDescriptions() : GetDescription(field);            });        }        /// <summary>        /// 獲取位域枚舉的描述,多個按分隔符組合        /// </summary>        public static string GetDescriptions(this Enum @this, string separator = ",")        {            var names = @this.ToString().Split(',');            string[] res = new string[names.Length];            var type = @this.GetType();            for (int i = 0; i < names.Length; i++)            {                var field = type.GetField(names[i].Trim());                if (field == null) continue;                res[i] = GetDescription(field);            }            return string.Join(separator, res);        }        private static string GetDescription(FieldInfo field)        {            var att = System.Attribute.GetCustomAttribute(field, typeof(DescriptionAttribute), false);            return att == null ? field.Name : ((DescriptionAttribute)att).Description;        }        /****************** test methods ******************/        public static string GetDescriptionOriginal(this Enum @this)        {            var name = @this.ToString();            var field = @this.GetType().GetField(name);            if (field == null) return name;            var att = System.Attribute.GetCustomAttribute(field, typeof(DescriptionAttribute), false);            return att == null ? field.Name : ((DescriptionAttribute)att).Description;        }        private static Dictionary<Enum, string> _LockDictionary = new Dictionary<Enum, string>();        public static string GetDescriptionByDictionaryWithLocak(this Enum @this)        {            if (_LockDictionary.ContainsKey(@this)) return _LockDictionary[@this];            Monitor.Enter(_obj);            if (!_LockDictionary.ContainsKey(@this))            {                var value = @this.GetDescriptionOriginal();                _LockDictionary.Add(@this, value);            }            Monitor.Exit(_obj);            return _LockDictionary[@this];        }        private static Dictionary<Enum, string> _ExceptionDictionary = new Dictionary<Enum, string>();        public static string GetDescriptionByDictionaryWithException(this Enum @this)        {            try            {                return _ExceptionDictionary[@this];            }            catch (KeyNotFoundException)            {                Monitor.Enter(_obj);                if (!_ExceptionDictionary.ContainsKey(@this))                {                    var value = @this.GetDescriptionOriginal();                    _ExceptionDictionary.Add(@this, value);                }                Monitor.Exit(_obj);                return _ExceptionDictionary[@this];            }        }        public static object _obj = new object();        private static ConcurrentDictionary<Enum, string> _ConcurrentDictionary = new ConcurrentDictionary<Enum, string>();        public static string GetDescriptionByConcurrentDictionary(this Enum @this)        {            return _ConcurrentDictionary.GetOrAdd(@this, (key) =>            {                var type = key.GetType();                var field = type.GetField(key.ToString());                return field == null ? key.ToString() : GetDescription(field);            });        }    }
View Code

2.測試類EnumTest.cs代碼: 

    [TestFixture]    public class EnumTest    {        [Test]        public void SimpleTest()        {            Console.WriteLine(EnumGender.Female.GetDescriptionOriginal());            Console.WriteLine(EnumGender.Male.GetDescriptionOriginal());            Console.WriteLine(EnumGender.Other.GetDescriptionOriginal());            var t1 = EnumGender.Male | EnumGender.Female;            Console.WriteLine((t1 & EnumGender.Male) == EnumGender.Male);            Console.WriteLine(t1 & ~EnumGender.Male);            Console.WriteLine(Enum.IsDefined(typeof(EnumGender), 0));        }        [Test]        public void GetDescriptionOriginal_Test()        {            var enums = this.GetTestEnums();            Console.WriteLine(enums.Count);            TestHelper.InvokeAndWriteAll(() =>            {                System.Threading.Tasks.Parallel.For(0, 1000000, (i, obj) =>                {                    foreach (var item in enums)                    {                        var a = item.GetDescriptionOriginal();                    }                });            });        }        [Test]        public void GetDescriptionByDictionaryWithLocak_Test()        {            var enums = this.GetTestEnums();            Console.WriteLine(enums.Count);            TestHelper.InvokeAndWriteAll(() =>            {                System.Threading.Tasks.Parallel.For(0, 1000000, (i, obj) =>                {                    foreach (var item in enums)                    {                        var a = item.GetDescriptionByDictionaryWithLocak();                    }                });            });        }        [Test]        public void GetDescriptionByDictionaryWithException_Test()        {            var enums = this.GetTestEnums();            Console.WriteLine(enums.Count);            TestHelper.InvokeAndWriteAll(() =>            {                System.Threading.Tasks.Parallel.For(0, 1000000, (i, obj) =>                {                    foreach (var item in enums)                    {                        var a = item.GetDescriptionByDictionaryWithException();                    }                });            });        }        [Test]        public void GetDescriptionByConcurrentDictionary_Test()        {            var enums = this.GetTestEnums();            Console.WriteLine(enums.Count);            TestHelper.InvokeAndWriteAll(() =>            {                System.Threading.Tasks.Parallel.For(0, 1000000, (i, obj) =>                {                    foreach (var item in enums)                    {                        var a = item.GetDescriptionByConcurrentDictionary();                    }                });            });        }        private List<Enum> GetTestEnums()        {            List<Enum> res = new List<Enum>();            res.Add(EnumMutliFTest.T1);            res.Add(EnumMutliFTest.T2);            res.Add(EnumMutliFTest.T3);            res.Add(EnumMutliFTest.T4);            res.Add(EnumMutliFTest.T5);            res.Add(EnumMutliFTest.T6);            res.Add(EnumMutliFTest.T7);            res.Add(EnumMutliFTest.T8);            res.Add(EnumMutliFTest.T9);            res.Add(EnumMutliFTest.T10);            res.Add(EnumMutliFTest.T11);            res.Add(EnumMutliFTest.T12);            res.Add(EnumMutliFTest.T13);            res.Add(EnumMutliFTest.T14);            res.Add(EnumMutliFTest.T15);            res.Add(EnumMutliFTest.T16);            res.Add(EnumMutliFTest.T17);            res.Add(EnumMutliFTest.T18);            res.Add(EnumMutliFTest.T19);            res.Add(EnumMutliFTest.T20);            res.Add(EnumMutliFTest.T21);            res.Add(EnumMutliFTest.T22);            res.Add(EnumMutliFTest.T23);            res.Add(EnumMutliFTest.T24);            res.Add(EnumMutliFTest.T25);            res.Add(EnumMutliFTest.T26);            res.Add(EnumMutliFTest.T27);            res.Add(EnumMutliFTest.T28);            res.Add(EnumMutliFTest.T29);            res.Add(EnumMutliFTest.T30);            res.Add(EnumMutliFTest.T31);            res.Add(EnumMutliFTest.T32);            res.Add(EnumMutliFTest.T33);            res.Add(EnumMutliFTest.T34);            res.Add(EnumMutliFTest.T35);            res.Add(EnumMutliFTest.T36);            res.Add(EnumMutliFTest.T37);            res.Add(EnumMutliFTest.T38);            res.Add(EnumMutliFTest.T3);            res.Add(EnumMutliFTest.T18);            res.Add(EnumMutliFTest2.T21);            res.Add(EnumMutliFTest2.T22);            res.Add(EnumMutliFTest2.T23);            res.Add(EnumMutliFTest2.T24);            res.Add(EnumMutliFTest2.T25);            res.Add(EnumMutliFTest2.T26);            res.Add(EnumMutliFTest2.T27);            res.Add(EnumMutliFTest2.T28);            res.Add(EnumMutliFTest2.T29);            res.Add(EnumMutliFTest2.T210);            res.Add(EnumMutliFTest2.T211);            res.Add(EnumMutliFTest2.T212);            res.Add(EnumMutliFTest2.T213);            res.Add(EnumMutliFTest2.T214);            res.Add(EnumMutliFTest2.T215);            res.Add(EnumMutliFTest2.T216);            res.Add(EnumMutliFTest2.T217);            res.Add(EnumMutliFTest2.T218);            res.Add(EnumMutliFTest2.T219);            res.Add(EnumMutliFTest2.T220);            res.Add(EnumMutliFTest2.T221);            res.Add(EnumMutliFTest2.T222);            res.Add(EnumMutliFTest2.T223);            res.Add(EnumMutliFTest2.T224);            res.Add(EnumMutliFTest2.T225);            res.Add(EnumMutliFTest2.T226);            res.Add(EnumMutliFTest2.T227);            res.Add(EnumMutliFTest2.T228);            res.Add(EnumMutliFTest2.T229);            res.Add(EnumMutliFTest2.T230);            res.Add(EnumMutliFTest2.T231);            res.Add(EnumMutliFTest2.T232);            res.Add(EnumMutliFTest2.T233);            res.Add(EnumMutliFTest2.T234);            res.Add(EnumMutliFTest2.T235);            res.Add(EnumMutliFTest2.T236);            res.Add(EnumMutliFTest2.T237);            res.Add(EnumMutliFTest2.T238);            res.Add(EnumMutliFTest2.T23);            res.Add(EnumMutliFTest2.T218);            return res;        }        public enum EnumMutliFTest        {            [System.ComponentModel.Description("DT1")]            T1,            [System.ComponentModel.Description("DT2")]            T2,            [System.ComponentModel.Description("DT3")]            T3,            [System.ComponentModel.Description("DT4")]            T4,            [System.ComponentModel.Description("DT5")]            T5,            [System.ComponentModel.Description("DT6")]            T6,            [System.ComponentModel.Description("DT7")]            T7,            [System.ComponentModel.Description("DT8")]            T8,            [System.ComponentModel.Description("DT9")]            T9,            [System.ComponentModel.Description("DT10")]            T10,            [System.ComponentModel.Description("DT11")]            T11,            [System.ComponentModel.Description("DT12")]            T12,            [System.ComponentModel.Description("DT13")]            T13,            [System.ComponentModel.Description("DT14")]            T14,            [System.ComponentModel.Description("DT15")]            T15,            [System.ComponentModel.Description("DT16")]            T16,            [System.ComponentModel.Description("DT17")]            T17,            [System.ComponentModel.Description("DT18")]            T18,            [System.ComponentModel.Description("DT19")]            T19,            [System.ComponentModel.Description("DT20")]            T20,            [System.ComponentModel.Description("DT21")]            T21,            [System.ComponentModel.Description("DT22")]            T22,            [System.ComponentModel.Description("DT23")]            T23,            [System.ComponentModel.Description("DT24")]            T24,            [System.ComponentModel.Description("DT25")]            T25,            [System.ComponentModel.Description("DT26")]            T26,            [System.ComponentModel.Description("DT27")]            T27,            [System.ComponentModel.Description("DT28")]            T28,            [System.ComponentModel.Description("DT29")]            T29,            [System.ComponentModel.Description("DT30")]            T30,            [System.ComponentModel.Description("DT31")]            T31,            [System.ComponentModel.Description("DT32")]            T32,            [System.ComponentModel.Description("DT33")]            T33,            [System.ComponentModel.Description("DT34")]            T34,            [System.ComponentModel.Description("DT35")]            T35,            [System.ComponentModel.Description("DT36")]            T36,            [System.ComponentModel.Description("DT37")]            T37,            [System.ComponentModel.Description("DT38")]            T38,        }        public enum EnumMutliFTest2        {            [System.ComponentModel.Description("DT21")]            T21,            [System.ComponentModel.Description("DT22")]            T22,            [System.ComponentModel.Description("DT23")]            T23,            [System.ComponentModel.Description("DT24")]            T24,            [System.ComponentModel.Description("DT25")]            T25,            [System.ComponentModel.Description("DT26")]            T26,            [System.ComponentModel.Description("DT27")]            T27,            [System.ComponentModel.Description("DT28")]            T28,            [System.ComponentModel.Description("DT29")]            T29,            [System.ComponentModel.Description("DT210")]            T210,            [System.ComponentModel.Description("DT211")]            T211,            [System.ComponentModel.Description("DT212")]            T212,            [System.ComponentModel.Description("DT213")]            T213,            [System.ComponentModel.Description("DT214")]            T214,            [System.ComponentModel.Description("DT215")]            T215,            [System.ComponentModel.Description("DT216")]            T216,            [System.ComponentModel.Description("DT217")]            T217,            [System.ComponentModel.Description("DT218")]            T218,            [System.ComponentModel.Description("DT219")]            T219,            [System.ComponentModel.Description("DT220")]            T220,            [System.ComponentModel.Description("DT221")]            T221,            [System.ComponentModel.Description("DT222")]            T222,            [System.ComponentModel.Description("DT223")]            T223,            [System.ComponentModel.Description("DT224")]            T224,            [System.ComponentModel.Description("DT225")]            T225,            [System.ComponentModel.Description("DT226")]            T226,            [System.ComponentModel.Description("DT227")]            T227,            [System.ComponentModel.Description("DT228")]            T228,            [System.ComponentModel.Description("DT229")]            T229,            [System.ComponentModel.Description("DT230")]            T230,            [System.ComponentModel.Description("DT231")]            T231,            [System.ComponentModel.Description("DT232")]            T232,            [System.ComponentModel.Description("DT233")]            T233,            [System.ComponentModel.Description("DT234")]            T234,            [System.ComponentModel.Description("DT235")]            T235,            [System.ComponentModel.Description("DT236")]            T236,            [System.ComponentModel.Description("DT237")]            T237,            [System.ComponentModel.Description("DT238")]            T238,        }        //['d?end?]        [Flags]        public enum EnumGender        {            None,            [System.ComponentModel.Description("")]            Male,            [System.ComponentModel.Description("")]            Female,            Other,        }    }
View Code

3.輔助測試類TestHelper.cs

    public static class TestHelper    {        /// <summary>        /// 執行一個方法并返回執行時間間隔        /// </summary>        public static TimeSpan InvokeAndGetTimeSpan(Action call)        {            Stopwatch sw = new Stopwatch();            sw.Start();            call();            sw.Stop();            return sw.Elapsed;        }        /// <summary>        /// 執行一個方法并Console輸出實際執行間隔(豪秒)        /// </summary>        [Conditional("DEBUG")]        public static void InvokeAndWriteTimeSpan(Action call)        {            Console.WriteLine("TimeSpan:{0:N4}ms", InvokeAndGetTimeSpan(call).TotalMilliseconds);        }        /// <summary>        /// 執行一個方法并返回托管內存使用大小(可能內存回收會導致不準確)        /// </summary>        public static long InvokeAndGetMemoryUsed(Action call)        {            var start = GC.GetTotalMemory(false);            call();            return GC.GetTotalMemory(false) - start;        }        /// <summary>        /// 執行一個方法并Console輸出托管內存使用大小(字節)        /// </summary>        [Conditional("DEBUG")]        public static void InvokeAndWriteMemoryUsed(Action call)        {            Console.WriteLine("MemoryUsed:{0:N4}KB", InvokeAndGetMemoryUsed(call) / 1024F);        }        /// <summary>        /// 執行一個方法并Console輸出:實際執行間隔(豪秒);托管內存使用大小(可能內存回收會導致不準確)        /// </summary>        [Conditional("DEBUG")]        public static void InvokeAndWriteAll(Action call)        {            Stopwatch sw = new Stopwatch();            sw.Start();            var start = GC.GetTotalMemory(false);            call();            var end = GC.GetTotalMemory(false);            sw.Stop();            Console.WriteLine("TimeSpan:{0:N4}ms", sw.ElapsedMilliseconds);            Console.WriteLine("MemoryUsed:{0:N4}KB", (end - start) / 1024F);            Console.WriteLine("CollectionCount(0):{0:N}", GC.CollectionCount(0));        }    }
View Code

 


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产日本欧美一区二区三区| 久久久中精品2020中文| 日韩中文字幕精品| 欧美日韩成人在线视频| 中文字幕在线视频日韩| 国产精品久久婷婷六月丁香| 欧美丰满少妇xxxxx| 国产精品一区久久| 亚洲人午夜精品免费| 欧美成年人视频网站欧美| 国产精品美女久久| 国产婷婷色综合av蜜臀av| 亚洲最大的成人网| 成人黄色在线观看| 97精品国产97久久久久久免费| 日韩综合中文字幕| 亚洲成在人线av| 欧美专区第一页| 欧美日韩免费看| 亚洲欧美中文字幕在线一区| 中文字幕亚洲情99在线| 精品伊人久久97| 久久成人这里只有精品| 日韩av三级在线观看| 亚洲欧美成人精品| 欧美国产精品人人做人人爱| 日韩精品在线视频观看| 秋霞成人午夜鲁丝一区二区三区| 久久视频免费在线播放| 韩国三级电影久久久久久| 欧美大片在线看免费观看| 日韩亚洲在线观看| 欧美成人久久久| 久久久久久久影视| 久久99精品久久久久久噜噜| 国产精品一区二区女厕厕| www欧美日韩| 国产精品影片在线观看| 久精品免费视频| 国产精品一二区| 91人人爽人人爽人人精88v| 欧美不卡视频一区发布| 亚洲一区av在线播放| 国产精品久久久久久久久久久久| 国内成人精品视频| 91超碰中文字幕久久精品| 欧美性视频精品| 日韩激情av在线免费观看| 欧美电影在线观看| 亚洲人成伊人成综合网久久久| 亚洲国产精品va在线看黑人动漫| 久久精品国产亚洲7777| 欧美一区二区三区艳史| 91亚洲精品久久久久久久久久久久| 国产精品人成电影在线观看| 国产裸体写真av一区二区| 久久久久久久久久国产| 久久久久久午夜| 久久琪琪电影院| 欧美有码在线视频| 中日韩美女免费视频网站在线观看| 日韩欧美亚洲一二三区| www国产精品com| www.国产一区| 日本精品一区二区三区在线| 中文字幕亚洲无线码在线一区| 日韩电影大片中文字幕| 亚洲精品乱码久久久久久金桔影视| 92福利视频午夜1000合集在线观看| 国产精品久久久久久久久久三级| 欧美日韩成人网| 久久91亚洲人成电影网站| 国产+人+亚洲| 欧美老女人在线视频| 亚洲国产中文字幕久久网| 亚洲精品二三区| 在线国产精品视频| 国产不卡一区二区在线播放| 自拍偷拍亚洲一区| 日韩av一区二区在线观看| 久久综合免费视频| 久久亚洲综合国产精品99麻豆精品福利| 97国产精品久久| 欧美国产高跟鞋裸体秀xxxhd| 欧美电影在线观看网站| 日韩免费av片在线观看| 亚洲电影av在线| 欧美黑人一区二区三区| 国产精品18久久久久久麻辣| 亚洲欧洲国产精品| 亚洲剧情一区二区| 亚洲综合中文字幕在线观看| 久久久久国产精品免费| 九九九久久国产免费| 欧美激情在线一区| 国产亚洲精品久久久久久牛牛| 91精品在线观| 一区二区福利视频| 亚洲综合精品一区二区| 欧美大片在线影院| 狠狠色狠狠色综合日日小说| 51精品国产黑色丝袜高跟鞋| 中文字幕亚洲专区| 亚洲欧美国产va在线影院| 亚洲а∨天堂久久精品喷水| 欧美午夜性色大片在线观看| 欧美日韩高清在线观看| 国产精品成人va在线观看| 久久精品视频导航| 亚洲最大av网站| 91国产高清在线| 中文字幕亚洲国产| 欧美黑人一级爽快片淫片高清| 久久久爽爽爽美女图片| 91黑丝在线观看| 456国产精品| 成人国产精品色哟哟| 亚洲人成毛片在线播放| 久久亚洲春色中文字幕| 亚洲欧美一区二区三区在线| 欧美成年人视频网站欧美| 亚洲男人天堂网站| 日韩精品视频免费在线观看| 上原亚衣av一区二区三区| 精品久久久久久亚洲精品| 欧美中文字幕视频在线观看| 国产精品久久久久久av下载红粉| 久久久久久久久久久91| 亚洲老板91色精品久久| 久久国产精品影片| 日韩视频免费在线观看| 中文字幕在线国产精品| 91影院在线免费观看视频| 久久久久久久国产精品视频| 亚洲国产精品久久91精品| 欧美精品久久久久久久久久| 国产精品扒开腿爽爽爽视频| 国产日韩欧美在线视频观看| 久久久最新网址| 91精品久久久久久久久久入口| 国产一区二区三区高清在线观看| 精品亚洲一区二区三区在线观看| 国语自产在线不卡| 欧美精品久久久久久久免费观看| 精品国产91乱高清在线观看| 91免费看视频.| 久久精品中文字幕电影| 欧美成人精品在线播放| 精品久久久香蕉免费精品视频| 国产精品福利在线观看网址| 久久国产精品久久久久久| 精品国产电影一区| 亚洲国产成人爱av在线播放| 萌白酱国产一区二区| 精品久久久久久国产91| 国产视频精品va久久久久久| 亚洲人永久免费| 亚洲第一精品自拍| 26uuu日韩精品一区二区| 欧美性高潮床叫视频| 在线丨暗呦小u女国产精品| 亚洲精品久久7777777| 日本午夜在线亚洲.国产| 日韩av在线免费看|