C#基礎(chǔ)之C#代碼的注意事項(上)
關(guān)于代碼優(yōu)化的問題,之前也給大家介紹過相關(guān)的內(nèi)容。下面介紹的是C#代碼優(yōu)化的一些注意事項,供參考。
一、用屬性代替可訪問的字段
1、.NET數(shù)據(jù)綁定只支持?jǐn)?shù)據(jù)綁定,使用屬性可以獲得數(shù)據(jù)綁定的好處;
2、在屬性的get和set訪問器重可使用lock添加多線程的支持。
二、readonly(運(yùn)行時常量)和const(編譯時常量)
1、const只可用于基元類型、枚舉、字符串,而readonly則可以是任何的類型;
2、const在編譯時將替換成具體的常量,這樣如果在引用中同時使用了const和readonly兩種值,則對readonly的再次改變將會改變設(shè)計的初衷,這是需要重新編譯所更改的程序集,以重新引用新的常量值。
3、const比readonly效率高,但失去了應(yīng)用的靈活性。
三、is與as
1、兩者都是在運(yùn)行時進(jìn)行類型的轉(zhuǎn)換,as操作符只能使用在引用類型,而is可以使用值和引用類型;
2、通常的做法是用is判斷類型,然后選擇使用as或強(qiáng)類型轉(zhuǎn)換操作符(用operater定義的轉(zhuǎn)換)有選擇地進(jìn)行。
四、ConditionalAttribute代替#if #endif條件編譯
1、ConditionalAttribute只用于方法級,對其他的如類型、屬性等的添加都是無效的;而#if #endif則不受此限制;
2、ConditionalAttribute可以添加多個編譯條件的或(OR)操作,而#if #endif則可以添加與(AND)[這里可以完全定義為另一個單獨(dú)的符號];
3、ConditioanlAttribute定義可以放在一個單獨(dú)的方法中,使得程序更為靈活。
五、提供ToString()方法
1、可以更友好的方式提供用戶詳細(xì)的信息;
2、使用IFormatter.ToString()方法提供更靈活的定制,如果添加IFormatProvider 和ICustomFormatter接口則更有意義的定制消息輸出。
六、值和引用類型的區(qū)別
1、值類型不支持多態(tài),適合存儲應(yīng)用程序操作的數(shù)據(jù),而引用則支持多態(tài),適用于定義應(yīng)用程序的行為;
2、對于數(shù)組定義為值類型可以顯著提高程序的性能;
3、值類型具有較少的堆內(nèi)存碎片、內(nèi)存垃圾和間接訪問時間,其在方法中的返回是以復(fù)制的方式進(jìn)行,避免暴露內(nèi)部結(jié)構(gòu)到外界;
4、值類型應(yīng)用在如下的場景中:類型的職責(zé)主要是用于數(shù)據(jù)存儲;公共接口完全由一些數(shù)據(jù)成員存取屬性定義;永遠(yuǎn)沒有子類;永遠(yuǎn)沒有多態(tài)行為。
七、值類型盡可能實(shí)現(xiàn)為常量性和原子性的類型
1、使我們的代碼更易于編寫和維護(hù);
2、初始化常量的三種策略:在構(gòu)造中;工廠方法;構(gòu)造一個可變的輔助類(如StringBuilder)。
八、確保0為值得有效狀態(tài)
1、值類型的默認(rèn)狀態(tài)應(yīng)為0;
2、枚舉類型的0不應(yīng)為“無效的狀態(tài)”;在FlagsAttribute是應(yīng)確保0值為有效地狀態(tài);
3、在字符串為為空時可以返回一個string.Empty的空字符串;
九、相等判斷的多種表示關(guān)系
1、ReferenceEquals()判斷引用相等,需要兩個是引用同一個對象時方可返回true;
2、靜態(tài)的Equals()方法先進(jìn)性引用判斷再進(jìn)行值類型判斷的;
3、對于引用類型的判斷可以在使用“值語義”時使用重寫Equals()方法;
4、重寫Equals()方法時也應(yīng)當(dāng)重寫GetHashCode()方法,同時提供operater==()操作。
十、理解GetHashCode()方法的缺陷
1、GetHashCode()僅應(yīng)用在基于散列的集合定義鍵的散列值,如HashTable或Dictionary;
2、GetHashCode()應(yīng)當(dāng)遵循相應(yīng)的三條規(guī)則:兩個相等對象應(yīng)當(dāng)返回相同的散列碼;應(yīng)當(dāng)是一個實(shí)例不變式;散列函數(shù)應(yīng)該在所有的整數(shù)中產(chǎn)生一個隨機(jī)的分布;
十一、優(yōu)先使用foreach循環(huán)語句
1、foreach可以消除編譯器對for循環(huán)對數(shù)組邊界的檢查;
2、foreach的循環(huán)變量是只讀的,且存在一個顯式的轉(zhuǎn)換,在集合對象的對象類型不正確時拋出異常;
3、foreach使用的集合需要有:具備公有的GetEnumberator()方法;顯式實(shí)現(xiàn)了IEnumberable接口;實(shí)現(xiàn)了IEnumerator接口;
4、foreach可以帶來資源管理的好處,因為如果編譯器可以確定IDisposable接口時可以使用優(yōu)化的try…finally塊;
十二、默認(rèn)字段的初始化優(yōu)于賦值語句
1、字段生命默認(rèn)會將值類型初始化為0,引用類型初始化為null;
2、對同一個對象進(jìn)行多次初始化會降低代碼的執(zhí)行效率;
3、將字段的初始化放到構(gòu)造器中有利于進(jìn)行異常處理。
十三、使用靜態(tài)構(gòu)造器初始化靜態(tài)成員
1、靜態(tài)構(gòu)造器會在一個類的任何方法、變量或者屬性訪問之前執(zhí)行;
2、靜態(tài)字段同樣會在靜態(tài)構(gòu)造器之前運(yùn)行,同時靜態(tài)構(gòu)造器有利于異常處理。
十四、利用構(gòu)造器鏈(在.NET 4.0已經(jīng)用可選參數(shù)解決了這個問題)
1、用this將初始化工作交給另一個構(gòu)造器,用base調(diào)用基類的構(gòu)造器;
2、類型實(shí)例的操作順序是:將所有的靜態(tài)字段都設(shè)置為0;執(zhí)行靜態(tài)字段初始化器;執(zhí)行基類的靜態(tài)構(gòu)造器;執(zhí)行當(dāng)前類型的靜態(tài)構(gòu)造器;
將所有的實(shí)例字段設(shè)置為0;執(zhí)行實(shí)例字段初始化器;執(zhí)行合適的基類實(shí)例構(gòu)造器;執(zhí)行當(dāng)前類型的實(shí)例構(gòu)造器。
十五、利用using和try/finally語句來清理資源
在IDisposable接口的Dispose()方法中用GC.SuppressFinalize()可通知垃圾收集器不再執(zhí)行終結(jié)操作。
十六、盡量減少內(nèi)存垃圾
1、分配和銷毀一個對上的對象都要花費(fèi)額外的處理器時間;
2、減少分配對象數(shù)量的技巧:經(jīng)常使用的局部變量提升為字段;提供一個類,用于存儲Singleton對象來表達(dá)特定類型的常用實(shí)例。
3、用StringBuilder進(jìn)行復(fù)雜的字符串操作。
十七、盡量減少裝箱和拆箱
1、關(guān)注一個類型到System.Object的隱式轉(zhuǎn)換,同時值類型不應(yīng)該被替換為System.Object類型;
2、使用接口而不是使用類型可以避免裝箱,即將值類型從接口實(shí)現(xiàn),然后通過接口調(diào)用成員。
十八、實(shí)現(xiàn)標(biāo)準(zhǔn)Dispose模式
1、使用非內(nèi)存資源,它必須有一個終結(jié)器,垃圾收集器在完成沒有終結(jié)其的內(nèi)存對象后會將實(shí)現(xiàn)了終結(jié)器對象的添加到終結(jié)隊列中,然后垃圾收集器會啟動一個新的線程來運(yùn)行這些對象上的終結(jié)器,這種防御性的變成方式是因為如果用戶忘記了調(diào)用Dispose()方法,垃圾回收器總是會調(diào)用終結(jié)器方法的,這樣可以避免出現(xiàn)非托管的內(nèi)存資源不被釋放引起內(nèi)存泄漏的問題;l
2、使用IDisposable.Dispose()方法需要做四個方面的工作:釋放所有的非托管資源;釋放所有的托管資源;設(shè)置一個狀態(tài)標(biāo)記來表示是否已經(jīng)執(zhí)行了Dispose();調(diào)用GC.SuppressFinalize(this)取消對象的終結(jié)操作;
3、為需要多態(tài)的類型添加一個受保護(hù)的虛方法Dispose(),派生類通過重寫這個方法來釋放自己的任務(wù);
4、在需要IDisoposable接口的類型中,即使我們不需要一個終結(jié)器也應(yīng)該實(shí)現(xiàn)一個終結(jié)器。
十九、定義并實(shí)現(xiàn)接口優(yōu)于繼承類型
1、不相關(guān)的類型可以共同實(shí)現(xiàn)一個共同的接口,而且實(shí)現(xiàn)接口比繼承更容易;
2、接口比較穩(wěn)定,他將一組功能封裝在一個接口中,作為其他類型的實(shí)現(xiàn)合同,而基類則可以隨著時間的推移進(jìn)行擴(kuò)展。
二十、明辨接口實(shí)現(xiàn)和虛方法重寫
1、在基類中實(shí)現(xiàn)一個接口時,派生類需要使用new來隱藏對基類方法的使用;
2、可以將基類接口的方法申明為虛方法,然后再派生類中實(shí)現(xiàn)。
二十一、使用委托表達(dá)回調(diào)
1、委托對象本身不提供任何異常捕獲,所以任何的多播委托調(diào)用都會結(jié)束整個調(diào)用鏈;
2、通過顯示調(diào)用委托鏈上的每個委托目標(biāo)可以避免多播委托僅返回最后一個委托的輸出。
二十二、使用事件定義外部接口
1、應(yīng)當(dāng)聲明為共有的事件,讓編譯器為我們創(chuàng)建add和renmove方法;
2、使用System.ComponentModel.EventHandlerList容器來存儲各個事件處理器,在類型中包含大量事件時可以使用他來隱藏所有事件的復(fù)雜性。
二十三、避免返回內(nèi)部類對象的引用
1、由于值類型對象的訪問會創(chuàng)建一個該對象的副本,所以定義一個值類型的的屬性完全不會改變類型對象內(nèi)部的狀態(tài);
2、常量類型可以避免改變對象的狀態(tài);
3、定義接口將訪問限制在一個子集中從而最小化對對象內(nèi)部狀態(tài)的破壞;
4、定義一個包裝器對象來限制另一個對象的訪問;
5、希望客戶代碼更改內(nèi)部數(shù)據(jù)元素時可以實(shí)現(xiàn)Observer模式,以使對象可以對更改進(jìn)行校驗或相應(yīng)。
二十四、聲明式編程優(yōu)于命令式編程
可以避免在多個類似的手工編寫的算法中犯錯誤的可能性,并提供清晰和可讀的代碼。
二十五、盡可能將類型實(shí)現(xiàn)為可序列化的類型
1、類型表示的不是UI控件、窗口或者表單,都應(yīng)使類型支持序列化;
2、在添加了NonSerializedAttribute的反序列化的屬性時可以通過實(shí)現(xiàn)IDeserializationCallback的OnDeserialization()方法裝入默認(rèn)值;
3、在版本控制中可以使用ISerializable接口來進(jìn)行靈活的控制,同時提供一個序列化的構(gòu)造器來根據(jù)流中的數(shù)據(jù)初始化對象,在實(shí)現(xiàn)時還要求SerializationFormatter異常的許可。
4、如果需要創(chuàng)建派生類則需要提供一個掛鉤方法供派生類使用。
由于本文過長的原因,分為兩篇為大家介紹,請看下一篇>>
【編輯推薦】