真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

C#開(kāi)發(fā)者需要注意什么-創(chuàng)新互聯(lián)

小編給大家分享一下C#開(kāi)發(fā)者需要注意什么,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

創(chuàng)新互聯(lián)一直通過(guò)網(wǎng)站建設(shè)和網(wǎng)站營(yíng)銷幫助企業(yè)獲得更多客戶資源。 以"深度挖掘,量身打造,注重實(shí)效"的一站式服務(wù),以網(wǎng)站設(shè)計(jì)制作、網(wǎng)站制作、移動(dòng)互聯(lián)產(chǎn)品、全網(wǎng)營(yíng)銷推廣服務(wù)為核心業(yè)務(wù)。十載網(wǎng)站制作的經(jīng)驗(yàn),使用新網(wǎng)站建設(shè)技術(shù),全新開(kāi)發(fā)出的標(biāo)準(zhǔn)網(wǎng)站,不但價(jià)格便宜而且實(shí)用、靈活,特別適合中小公司網(wǎng)站制作。網(wǎng)站管理系統(tǒng)簡(jiǎn)單易用,維護(hù)方便,您可以完全操作網(wǎng)站資料,是中小公司快速網(wǎng)站建設(shè)的選擇。

1.開(kāi)發(fā)流程

程序的Bug與瑕疵往往出現(xiàn)于開(kāi)發(fā)流程當(dāng)中。只要對(duì)工具善加利用,就有助于在你發(fā)布程序之前便將問(wèn)題發(fā)現(xiàn),或避開(kāi)這些問(wèn)題。

標(biāo)準(zhǔn)化代碼書寫

標(biāo)準(zhǔn)化代碼書寫可以使代碼更加易于維護(hù),尤其是在代碼由多個(gè)開(kāi)發(fā)者或團(tuán)隊(duì)進(jìn)行開(kāi)發(fā)與維護(hù)時(shí),這一優(yōu)點(diǎn)更加突出。常見(jiàn)的強(qiáng)制代碼規(guī)范化的工具有:FxCop、StyleCop和ReSharper。

開(kāi)發(fā)者語(yǔ):在掩蓋錯(cuò)誤之前請(qǐng)仔細(xì)地思考這些錯(cuò)誤,并且去分析結(jié)果。不要指望依靠這些工具來(lái)在代碼中尋找錯(cuò)誤,因?yàn)榻Y(jié)果可能和你的與其相去甚遠(yuǎn)。

代碼審查

審查代碼與搭檔編程都是很常見(jiàn)的練習(xí),比如開(kāi)發(fā)者刻意去審查他人書寫的代碼。而其他人很希望發(fā)現(xiàn)代碼開(kāi)發(fā)者的一些bug,例如編碼錯(cuò)誤或者執(zhí)行錯(cuò)誤。

審查代碼是一種很有價(jià)值的練習(xí),由于很依賴于人工操作,因此很難被量化,準(zhǔn)確度也不夠令人滿意。

靜態(tài)分析

靜態(tài)分析不需要你去運(yùn)行代碼,你不必編寫測(cè)試案例就可以找出一些代碼不規(guī)范的地方,或者是一些瑕疵的存在。這是一種非常有效地尋找問(wèn)題的方式,但是你需要有一個(gè)不會(huì)有太多誤報(bào)問(wèn)題的工具。C#常用的靜態(tài)分析工具有Coverity,CAT,NET,Visual Studio Code Analysis。

動(dòng)態(tài)分析

在你運(yùn)行代碼的時(shí)候,動(dòng)態(tài)分析工具可以幫你找出這些錯(cuò)誤:安全漏洞,性能與并發(fā)性問(wèn)題。這種方法是在執(zhí)行時(shí)期的環(huán)境下進(jìn)行分析,正因如此,其有效性便受制于代碼復(fù)雜度。Visual Studio提供了包括Concurrency Visualizer, IntelliTrace, and Profiling Tools在內(nèi)的大量動(dòng)態(tài)分析工具。

管理者/團(tuán)隊(duì)領(lǐng)導(dǎo)語(yǔ):開(kāi)發(fā)實(shí)踐是練習(xí)規(guī)避常見(jiàn)陷阱的最好方法。同時(shí)也要注意測(cè)試工具是否符合你的需求。盡量讓你團(tuán)隊(duì)的代碼診斷水平處于可控的范圍內(nèi)。

測(cè)試

測(cè)試的方式多種多樣:?jiǎn)卧獪y(cè)試,系統(tǒng)集成測(cè)試,性能測(cè)試,滲透測(cè)試等等。在開(kāi)發(fā)階段,絕大多數(shù)的測(cè)試案例是由開(kāi)發(fā)者或測(cè)試人員來(lái)完成編寫,使程序可以滿足需求。

測(cè)試只在運(yùn)行正確的代碼時(shí)才會(huì)有效。在進(jìn)行功能測(cè)試的時(shí)候,它還可以用來(lái)挑戰(zhàn)開(kāi)發(fā)者的研發(fā)與維護(hù)速度。

開(kāi)發(fā)最佳實(shí)踐

工具的選擇上多花點(diǎn)時(shí)間,用正確的工具去解決你關(guān)心的問(wèn)題,不要為開(kāi)發(fā)者增添額外的工作。讓分析工具與測(cè)試自動(dòng)流暢地運(yùn)行起來(lái)去尋找問(wèn)題,但是要保證代碼的思想仍然清晰地留在開(kāi)發(fā)者的頭腦當(dāng)中。

盡可能快地定位診斷出來(lái)的問(wèn)題所在位置(不論是通過(guò)靜態(tài)分析還是測(cè)試得到的錯(cuò)誤,比如編譯警告,標(biāo)準(zhǔn)違例,問(wèn)題檢測(cè)等)。如果剛出來(lái)的問(wèn)題由于“不關(guān)心”而去忽略它,導(dǎo)致該問(wèn)題后來(lái)很難找到,那么就會(huì)給代碼審閱工作者增加很大的工作量,并且還要祈禱他們不會(huì)因此煩躁。

請(qǐng)接受這些有用的建議,讓自己代碼的質(zhì)量,安全性,可維護(hù)性得到提升,同時(shí)也提升開(kāi)發(fā)者們的研發(fā)能力、協(xié)調(diào)能力,以及提升發(fā)布代碼的可預(yù)測(cè)性。

目標(biāo) 工具 影響
一致性,可維護(hù)性 標(biāo)準(zhǔn)化代碼書寫,靜態(tài)分析,代碼審查 間距一致,命名標(biāo)準(zhǔn),良好的可讀格式,都會(huì)讓開(kāi)發(fā)者更易編寫與維護(hù)代碼。
準(zhǔn)確性 代碼審查,靜態(tài)分析,動(dòng)態(tài)分析,測(cè)試 代碼不只是需要語(yǔ)法正確,還需要以開(kāi)發(fā)者的思想來(lái)滿足軟件需求。
功能性 測(cè)試 測(cè)試可以驗(yàn)證大多數(shù)的需求是否得到滿足:正確性,可拓展性,魯棒性以及安全性。
安全性 標(biāo)準(zhǔn)化代碼書寫,代碼審查,靜態(tài)分析,動(dòng)態(tài)分析,測(cè)試 安全性是一個(gè)復(fù)雜的問(wèn)題,任何一個(gè)小的漏洞都是潛在的威脅。
開(kāi)發(fā)者研發(fā)能力 標(biāo)準(zhǔn)化代碼書寫,靜態(tài)分析,測(cè)試 開(kāi)發(fā)者在工具的幫助下會(huì)很快速地更正錯(cuò)誤。
發(fā)布可預(yù)測(cè)性 標(biāo)準(zhǔn)化代碼書寫,代碼審查,靜態(tài)分析,動(dòng)態(tài)分析,測(cè)試 流線型后期階段的活動(dòng)、最小化錯(cuò)誤定位循環(huán),都可以讓問(wèn)題發(fā)現(xiàn)的更早。

2.類型的陷阱

C#的一個(gè)主要的優(yōu)點(diǎn)就是其靈活的類型系統(tǒng),而安全的類型可以幫助我們更早地找到錯(cuò)誤。通過(guò)強(qiáng)制執(zhí)行嚴(yán)格的類型規(guī)則,編譯器能夠幫助你維持良好的代碼書寫習(xí)慣。在這一方面,C#語(yǔ)言與.NET框架為我們提供了大量的類型,以適應(yīng)絕大多數(shù)的需求。雖然許多開(kāi)發(fā)者對(duì)一般的類型有著良好的理解,并且也知曉用戶的需求,但是一些誤解與誤用仍然存在。

更多關(guān)于.NTE框架類庫(kù)的信息請(qǐng)參閱MSDN library。

理解并使用標(biāo)準(zhǔn)接口

特定的接口涉及到常用的C#特征。例如,IDiposable允許使用常見(jiàn)的資源管理語(yǔ)言,例如關(guān)鍵詞“using”。良好地理解接口可以幫助你書寫通順的C#代碼,并且更易于維護(hù)。

避免使用ICloneable接口——開(kāi)發(fā)者從來(lái)沒(méi)搞清楚一個(gè)被復(fù)制的對(duì)象到底是深拷貝還是淺拷貝。由于仍沒(méi)有一種對(duì)復(fù)制對(duì)象操作是否正確的標(biāo)準(zhǔn)評(píng)判,于是也就沒(méi)辦法有意義地去將接口作為一個(gè)contract去使用。

結(jié)構(gòu)體

盡量避免向結(jié)構(gòu)體中進(jìn)行寫入,將它們視為一種不變的對(duì)象以防止混亂。在像多線程這種場(chǎng)景下進(jìn)行內(nèi)存共享,會(huì)變得更安全。我們對(duì)結(jié)構(gòu)體采用的方法是,在創(chuàng)建結(jié)構(gòu)體時(shí)對(duì)其進(jìn)行初始化操作,如果需要改變其數(shù)據(jù),那么建議生成一個(gè)新的實(shí)體。

正確理解哪些標(biāo)準(zhǔn)類型/方法是不可變,并且可返回新的值(例如串,日期),用這些來(lái)替代那些易變對(duì)象(如List.Enumerator)。

字符串

字符串的值可能為空,所以可以在合適的時(shí)候使用一些比較方便的功能。值判斷(s.Length==0)時(shí)可能會(huì)出現(xiàn)NullReferenceException錯(cuò)誤,而String.IsNullOrEmpty(s)和String.IsNullOrWhitespace(s)可以很好地使用null。

標(biāo)記枚舉

枚舉類型與常量可以使代碼更加易于閱讀,通過(guò)利用標(biāo)識(shí)符替換幻數(shù),可以表現(xiàn)出值的意義。

如果你需要生成大量的枚舉類型,那么帶有標(biāo)記的枚舉類型是一種更加簡(jiǎn)單的選擇:

[Flag]
public enum Tag {
  None   =0x0,
  Tip    =0x1,
  Example=0x2
}

下面這種方法可以讓你在一個(gè)snippet中使用多重標(biāo)記:

snippet.Tag = Tag.Tip | Tag.Example

這種方法有利于數(shù)據(jù)的封裝,因此你也不必?fù)?dān)心在使用Tag property getter時(shí)有內(nèi)部集合信息泄露。

Equality comparisons(相等性比較)

有如下兩種類型的相等性:

1.引用相等性,即兩種引用都指向同一個(gè)對(duì)象。

2.數(shù)值相等性,即兩個(gè)不同的引用對(duì)象可以視為相等的。

除此之外,C#還提供了很多相等性的測(cè)試方法。最常見(jiàn)的方法如下:

  • ==與!=操作

  • 由對(duì)象的虛繼承等值法

  • 靜態(tài)Object.Equal法

  • IEquatable接口等值法

  • 靜態(tài)Object.ReferenceEquals法

有時(shí)候很難弄清楚使用引用或值相等性的目的。想進(jìn)一步弄明白這些,并且讓你的工作做得更好,請(qǐng)參閱:

MSDNhttp://msdn.microsoft.com/en-us/library/dd183752.aspx

如果你想要覆蓋某個(gè)東西的時(shí)候,不要忘了MSDN上為我們提供的諸如IEquatable, GetHashCode()之類的工具。

注意無(wú)類型容器在重載方面的影響,可以考慮使用“myArrayList[0] == myString”這一方法。數(shù)組元素是編譯階段類型的“對(duì)象”,因此引用相等性可以使用。雖然C#會(huì)向你提醒這些潛在的錯(cuò)誤,但是在編譯過(guò)程中,unexpected reference equality在某些情況下不會(huì)被提醒。

3.類的陷阱

封裝你的數(shù)據(jù)

類在恰當(dāng)管理數(shù)據(jù)方面起很大的作用。鑒于性能上的一些原因,類總是緩存部分結(jié)果,或者是在內(nèi)部數(shù)據(jù)的一致性上做出一些假設(shè)。使數(shù)據(jù)權(quán)限公開(kāi)的話會(huì)在一定程度上讓你去緩存,或者是作出假設(shè),而這些操作是通過(guò)對(duì)性能、安全性、并發(fā)性的潛在影響表現(xiàn)出來(lái)的。例如暴露像泛型集合、數(shù)組之類的易變成員項(xiàng),可以讓用戶跳過(guò)你而直接進(jìn)行結(jié)構(gòu)體的修改。

屬性

除了可以通過(guò)access modifiers控制對(duì)象之外,屬性還可以讓你很精確地掌控用戶與你的對(duì)象之間進(jìn)行了什么交互。特別要指出的是,屬性還可以讓你了解到讀寫的具體情況。

屬性能在通過(guò)存儲(chǔ)邏輯將數(shù)據(jù)覆寫進(jìn)getters與setters的時(shí)候幫助你建立一個(gè)穩(wěn)定的API,或是提供一個(gè)數(shù)據(jù)的綁定資源。

永遠(yuǎn)不要讓屬性getter出現(xiàn)異常,并且也要避免修改對(duì)象狀態(tài)。這是一種對(duì)方法的需求,而不是屬性的getter。

更多有關(guān)屬性的信息,請(qǐng)參閱MSDN:

http://msdn.microsoft.com/en-us/library/ms229006(v=vs.120).aspx

同時(shí)也要注意getter的一些副作用。開(kāi)發(fā)者也習(xí)慣于將成員體的存取視為一種常見(jiàn)的操作,因此他們?cè)诖a審查的時(shí)候也常常忽視那些副作用。

對(duì)象初始化

你可以為一個(gè)新創(chuàng)建的對(duì)象根據(jù)它創(chuàng)建的表達(dá)形式賦予屬性。例如為Foo與Bar屬性創(chuàng)建一個(gè)新的具有給定值的C類對(duì)象:

new C {Foo=blah, Bar=blam}

你也可以生成一個(gè)具有特定屬性名稱的匿名類型的實(shí)體:

var myAwesomeObject = new {Name=”Foo”, Size=10};

初始化過(guò)程在構(gòu)造函數(shù)體之前運(yùn)行,因此需要保證在輸入至構(gòu)造函數(shù)之前,將這一域給初始化。由于構(gòu)造函數(shù)還沒(méi)有運(yùn)行,所以目標(biāo)域的初始化可能不管怎樣都不涉及“this”。

過(guò)渡規(guī)范細(xì)化的輸入?yún)?shù)

為了使一些特殊方法更加容易控制,最好在你使用的方法當(dāng)中使用最少的特定類型。比如在一種方法中使用 List進(jìn)行迭代:

public void Foo(List bars) 
{
  foreach(var b in bars)
  {
    // do something with the bar...
  }
}

對(duì)于其他IEnumerable集來(lái)說(shuō),使用這種方法的表現(xiàn)更加出色一些,但是對(duì)于特定的參數(shù)List來(lái)說(shuō),我們更需要使集以表的形式表現(xiàn)。盡量少地選取特定的類型(諸如IEnumerable, ICollection此類)以保證你的方法效率的大化。

4.泛型

泛型是一種在定義獨(dú)立類型結(jié)構(gòu)體與設(shè)計(jì)算法上一種十分有力的工具,它可以強(qiáng)制類型變得安全。

用像List這樣的泛型集來(lái)替代數(shù)組列表這種無(wú)類型集,既可以提升安全性,又可以提升性能。

在使用泛型時(shí),我們可以用關(guān)鍵詞“default”來(lái)為類型獲取缺省值(這些缺省值不可以硬編碼寫進(jìn)implementation)。特別要指出的是,數(shù)字類型的缺省值是o,引用類型與空類型的缺省值為null。

T t = default(T);

5.類型轉(zhuǎn)換

類型轉(zhuǎn)換有兩種模式。其一顯式轉(zhuǎn)換必須由開(kāi)發(fā)者調(diào)用,另一隱式轉(zhuǎn)換是基于環(huán)境下應(yīng)用于編譯器的。

常量o可由隱式轉(zhuǎn)換至枚舉型數(shù)據(jù)。當(dāng)你嘗試調(diào)用含有數(shù)字的方法時(shí),可以將這些數(shù)據(jù)轉(zhuǎn)換成枚舉類型。

類型轉(zhuǎn)換 描述
Tree tree = (Tree)obj; 這種方法可以在對(duì)象是樹類型時(shí)使用;如果對(duì)象不是樹,可能會(huì)出現(xiàn)InvalidCast異常。
Tree tree = obj as Tree; 這種方法你可以在預(yù)測(cè)對(duì)象是否為樹時(shí)使用。如果對(duì)象不是樹,那么會(huì)給樹賦值null。你可以用“as”的轉(zhuǎn)換,然后找到null值的返回處,再進(jìn)行處理。由于它需要有條件處理的返回值,因此記住只在需要的時(shí)候才去用這種轉(zhuǎn)換。這種額外的代碼可能會(huì)造成一些bug,還可能會(huì)降低代碼的可讀性。

轉(zhuǎn)換通常意味著以下兩件事之一:

1.RuntimeType的表現(xiàn)可比編譯器所表現(xiàn)出來(lái)的特殊的多,Cast轉(zhuǎn)換命令編譯器將這種表達(dá)視為一種更特殊的類型。如果你的設(shè)想不正確的話,那么編譯器會(huì)向你輸出一個(gè)異常。例如:將對(duì)象轉(zhuǎn)換成串。

2.有一種完全不同的類型的值,與Expression的值有關(guān)。Cast命令編譯器生成代碼去與該值相關(guān)聯(lián),或者是在沒(méi)有值的情況下報(bào)出一個(gè)異常。例如:將double類型轉(zhuǎn)換成int類型。

以上兩種類型的Cast都有著風(fēng)險(xiǎn)。第一種Cast向我們提出了一個(gè)問(wèn)題:“為什么開(kāi)發(fā)者能很清楚地知道問(wèn)題,而編譯器為什么不能?”如果你處于這個(gè)情況當(dāng)中,你可以去嘗試改變程序讓編譯器能夠順利地推理出正確的類型。如果你認(rèn)為一個(gè)對(duì)象的runtime type是比compile time type還要特殊的類型,你就可以用“as”或者“is”操作。

第二種cast也提出了一個(gè)問(wèn)題:“為什么不在第一步就對(duì)目標(biāo)數(shù)據(jù)類型進(jìn)行操作?”如果你需要int類型的結(jié)果,那么用int會(huì)比double更有意義一些。

獲取額外的信息請(qǐng)參閱:

http://blogs.msdn.com/b/ericlippert/archive/tags/cast+operator/

在某些情況下顯式轉(zhuǎn)換是一種正確的選擇,它可以提高代碼可閱讀性與debug能力,還可以在采用合適的操作的情況下提高測(cè)試能力。

6.異常

異常并不是condition

異常不應(yīng)該常出現(xiàn)在程序流程中。它們代表著開(kāi)發(fā)者所不愿看到的運(yùn)行環(huán)境,而這些很可能無(wú)法修復(fù)。如果你期望得到一個(gè)可控制的環(huán)境,那么主動(dòng)去檢查環(huán)境會(huì)比等待問(wèn)題的出現(xiàn)要好得多。

利用TryParse()方法可以很方便地將格式化的串轉(zhuǎn)換成數(shù)字。不論是否解析成功,它都會(huì)返回一個(gè)布爾型結(jié)果,這要比單純返回異常要好很多。

注意使用exception handling scope

寫代碼時(shí)注意catch與finally塊的使用。由于這些不希望得到的異常,控制可能進(jìn)入這些塊中。那些你期望的已執(zhí)行的代碼可能會(huì)由于異常而跳過(guò)。如:

Frobber originalFrobber = null;
try {
  originalFrobber = this.GetCurrentFrobber();
  this.UseTemporaryFrobber();
  this.frobSomeBlobs();
}
finally {
  this.ResetFrobber(originalFrobber);
}

如果GetCurrentFrobber()報(bào)出了一個(gè)異常,那么當(dāng)finally blocks被執(zhí)行時(shí)originalFrobber的值仍然為空。如果GetCurrentFrobber不能被扔掉,那么為什么其內(nèi)部是一個(gè)try block?

明智地處理異常

要注意有針對(duì)性地處理你的目標(biāo)異常,并且只去處理目標(biāo)代碼當(dāng)中的異常部分。盡量不要去處理所有異常,或者是根類異常,除非你的目的是記錄并重新處理這些異常。某些異常會(huì)使應(yīng)用處于一種接近崩潰的狀態(tài),但這也比無(wú)法修復(fù)要好得多。有些試圖修復(fù)代碼的操作可能會(huì)誤使情況變得更糟糕。

關(guān)于致命的異常都有一些細(xì)微的差異,特別是注重finally blocks的執(zhí)行,可以影響到異常的安全與調(diào)試。更多信息請(qǐng)參閱:

http://incrediblejourneysintotheknown.blogspot.com/2009/02/fatal-exceptions-and-why-vbnet-has.html

使用一款頂級(jí)的異常處理器去安全地處理異常情況,并且會(huì)將debug的一些問(wèn)題信息暴露出來(lái)。使用catch塊會(huì)比較安全地定位那些特殊的情況,從而安全地解決這些問(wèn)題,再將一些問(wèn)題留給頂級(jí)的異常處理器去解決。

如果你發(fā)現(xiàn)了一個(gè)異常,請(qǐng)做些什么去解決它,而不要去將這個(gè)問(wèn)題擱置。擱置只會(huì)使問(wèn)題更加復(fù)雜,更難以解決。

將異常包含至一個(gè)自定義異常中,對(duì)面向公共API的代碼特別有用。異常是可視界面方法的一部分,它也被參數(shù)與返回值所控制。但這種擴(kuò)散了很多異常的方法對(duì)于代碼的魯棒性與可維護(hù)性的解決來(lái)說(shuō)十分麻煩。

拋出(Throw)與繼續(xù)拋出(ReThrow)異常

如果你希望在更高層次上解決caught異常,那么就維持原異常狀態(tài),并且棧就是一個(gè)很好的debug方法。但需要注意維持好debug與安全考慮的平衡。

好的選擇包括簡(jiǎn)單地將異常繼續(xù)拋出:

Throw;

或者將異常視為內(nèi)部異常重新拋出:

拋出一個(gè)新CustomException;

不要顯式重新拋出類似于這樣的caught異常:

Throw e;

這樣的話會(huì)將異常的處理恢復(fù)至初始狀態(tài),并且阻礙debug。

有些異常發(fā)生于你代碼的運(yùn)行環(huán)境之外。與其使用caught塊,你可能更需要向目標(biāo)當(dāng)中添加如ThreadException或UnhandledException之類的處理器。例如,Windows窗體異常并不是出現(xiàn)于窗體處理線程環(huán)境當(dāng)中的。

原子性(數(shù)據(jù)完整性)

千萬(wàn)不要讓異常影響到你數(shù)據(jù)模型的完整性。你需要保證你的對(duì)象處于比較穩(wěn)定的狀態(tài)當(dāng)中——這樣一來(lái)任何由類的執(zhí)行的操作都不會(huì)出現(xiàn)違例。否則,通過(guò)“恢復(fù)”這一手段會(huì)使你的代碼變得更加讓人不解,也容易造成進(jìn)一步的損壞。

考慮幾種修改私有域順序的方法。如果在修改順序的過(guò)程當(dāng)中出現(xiàn)了異常,那么你的對(duì)象可能并不處于非法狀態(tài)下。嘗試在實(shí)際更新域之前去得到新的值,這樣你就可以在異常安全管理下,正常地更新你的域。

對(duì)特定類型的值——包括布爾型,32bit或者更小的數(shù)據(jù)類型與引用型——進(jìn)行可變量的分配,確??梢允窃有汀](méi)有什么保障是給一些大型數(shù)據(jù)(double,long,decimal)使用的。可以多考慮這個(gè):在共享多線程的變量時(shí),多使用lock statements。

7.事件

事件與委托共同提供了一種關(guān)于類的方法,這種方法在有特殊的事情發(fā)生時(shí)向用戶進(jìn)行提醒。委托事件的值在事件發(fā)生時(shí)應(yīng)被調(diào)用。事件就像是委托類型的域,當(dāng)對(duì)象生成時(shí),其自動(dòng)初始化為null。

事件也像值為“組播”的域。這也就是說(shuō),一種委托可以依次調(diào)用其他委托。你可以將一個(gè)委托分配給一個(gè)事件,你也可以通過(guò)類似-=于+=這樣的操作來(lái)控制事件。

注意資源競(jìng)爭(zhēng)

如果一個(gè)事件被多個(gè)線程所共享,另一個(gè)線程就有可能在你檢查是否為null之后,在調(diào)用其之前而清除所有的用戶信息——并拋出一個(gè)NullReferenceException。

對(duì)于此類問(wèn)題的標(biāo)準(zhǔn)解決方法是創(chuàng)建一個(gè)該事件的副本,用于測(cè)試與調(diào)用。你仍然需要注意的是,如果委托沒(méi)有被正確調(diào)用的話,那么在其他線程里被移除的用戶仍然可以繼續(xù)操作。你也可以用某種方法將操作按順序鎖定,以避免一些問(wèn)題。

public event EventHandler SomethingHappened;
private void OnSomethingHappened()
{
  // The event is null until somebody hooks up to it
  // Create our own copy of the event to protect against another thread removing our subscribers
  EventHandler handler = SomethingHappened;
  if (handler != null)
    handler(this,new EventArgs());
}

更多關(guān)于事件與競(jìng)爭(zhēng)的信息請(qǐng)參閱:

http://blogs.msdn.com/b/ericlippert/archive/2009/04/29/events-and-races.aspx

不要忘記將事件處理器Unhook

使用一種事件處理器為事件資源生成一個(gè)由處理器的資源對(duì)象到接收對(duì)象的引用,可以保護(hù)接收端的garbage collection。

適當(dāng)?shù)膗nhook處理器可以確保你不必因委托不再工作而去調(diào)用它浪費(fèi)時(shí)間,也不會(huì)使內(nèi)存存儲(chǔ)無(wú)用委托與不可引用的對(duì)象。

8.屬性

屬性提供了一種向程序集、類與其信息屬性中注入元數(shù)據(jù)的方法。它們經(jīng)常用來(lái)提供信息給代碼的消費(fèi)者——比如debugger、框架測(cè)試、應(yīng)用——通過(guò)反射這一方式。你也可以向你的用戶定義屬性,或是使用預(yù)定義屬性,詳見(jiàn)下表:

屬性 使用對(duì)象 目的
DebuggerDisplay Debugger Debugger display 格式
InternalsVisibleTo Member access 使用特定類來(lái)暴露內(nèi)部成員去指定其他的類。基于此方法,測(cè)試方法可以用來(lái)保護(hù)成員,并且persistence層可以用一些特殊的隱蔽方法。
DefaultValue Properties 為屬性指定一個(gè)缺省值

一定要對(duì)DebuggerStepThrough多重視幾分——否則它會(huì)在這個(gè)方法應(yīng)用的地方讓尋找bug變得十分困難,你也會(huì)因此而跳過(guò)某步或是推倒而重做它。

9.Debug

Debug是在開(kāi)發(fā)過(guò)程中必不可少的部分。除了使運(yùn)行環(huán)境不透明的部分變得可視化之外,debugger也可以侵入運(yùn)行環(huán)境,并且如果不使用debugger的話會(huì)導(dǎo)致應(yīng)用程序變現(xiàn)有所不同。

使異常??梢暬?/h4>

為了觀察當(dāng)前框架異常狀態(tài),你可以將“$exception”這一表達(dá)添加進(jìn)Visual Studio Watch窗口。這種變量包含了當(dāng)前異常狀態(tài),類似于你在catch block中所看見(jiàn)的,但其中不包含在debugger中看見(jiàn)的不是代碼中的真正存在的異常。

注意訪問(wèn)器的副作用

如果你的屬性有副作用,那么考慮你是否應(yīng)使用特性或者是debugger設(shè)置去避免debugger自動(dòng)地調(diào)用getter。例如,你的類可能有這樣一個(gè)屬性:

private int remainingAccesses = 10;
private string meteredData;
public string MeteredData
{
  get
  {
    if (remainingAccesses-- > 0)
      return meteredData;
    return null;
  }
}

你第一次在debugger中看見(jiàn)這個(gè)對(duì)象時(shí),remainingAccesses會(huì)獲得一個(gè)值為10的整型變量,并且MeteredData為null。然而如果你hover結(jié)束了remainingAccesses,你會(huì)發(fā)現(xiàn)它的值會(huì)變成9.這樣一來(lái)debugger的屬性值表現(xiàn)改變了你的對(duì)象的狀態(tài)。

10.性能優(yōu)化

早做計(jì)劃,不斷監(jiān)測(cè),后做優(yōu)化

在設(shè)計(jì)階段,制定切實(shí)可行的目標(biāo)。在開(kāi)發(fā)階段,專注于代碼的正確性要比去做微調(diào)整有意義的多。對(duì)于你的目標(biāo),你要在開(kāi)發(fā)過(guò)程中多進(jìn)行監(jiān)測(cè)。只需要在你沒(méi)有達(dá)到預(yù)期的目標(biāo)的時(shí)候,你才應(yīng)該去花時(shí)間對(duì)程序做一個(gè)調(diào)整。

請(qǐng)記住用合適的工具來(lái)確保性能的經(jīng)驗(yàn)性測(cè)量,并且使測(cè)試處于這樣一種環(huán)境當(dāng)中:可反復(fù)多次測(cè)試,并且測(cè)試過(guò)程盡量與現(xiàn)實(shí)當(dāng)中用戶的使用習(xí)慣一致。

當(dāng)你對(duì)性能進(jìn)行測(cè)試的時(shí)候,一定要注意你真正所關(guān)心的測(cè)試目標(biāo)是什么。在進(jìn)行某一項(xiàng)功能的測(cè)試時(shí),你的測(cè)試有沒(méi)有包含這項(xiàng)功能的調(diào)用或者是回路構(gòu)造的開(kāi)銷?

我們都聽(tīng)說(shuō)過(guò)很多比別人做得快很多的項(xiàng)目神話,不要盲目相信這些,試驗(yàn)與測(cè)試才是實(shí)在的東西。

由于CLR優(yōu)化的原因,有時(shí)候看起來(lái)效率不高的代碼可能會(huì)比看起來(lái)效率高的代碼運(yùn)行的更快。例如,CLR優(yōu)化循環(huán)覆蓋了一個(gè)完整的數(shù)組,以避免在不可見(jiàn)的per-element范圍里的檢查。開(kāi)發(fā)者經(jīng)常在循環(huán)一個(gè)數(shù)組之前先計(jì)算一下它的長(zhǎng)度:

int[] a_val = int[4000];
int len = a_val.Length;
for (int i = 0; i < len; i++)
    a_val[i] = i;

通過(guò)將長(zhǎng)度存儲(chǔ)進(jìn)一個(gè)變量當(dāng)中,CLR會(huì)不去識(shí)別這一部分,并且跳過(guò)優(yōu)化。但是有時(shí)手動(dòng)優(yōu)化會(huì)反人類地導(dǎo)致更糟糕的性能表現(xiàn)。

構(gòu)造字符串

如果你打算將大量的字符串進(jìn)行連接,可以使用System.Text.StringBuilder來(lái)避免生成大量的臨時(shí)字符串。

對(duì)集合使用批量處理

如果你打算生成并填滿集合中已知的大量數(shù)據(jù),由于再分配的存在,可以用保留空間來(lái)解決生成集合的性能與資源問(wèn)題。你可以用AddRange方法來(lái)進(jìn)一步對(duì)性能進(jìn)行優(yōu)化,如下在List中處理:

Persons.AddRange(listBox.Items);

11.資源管理

垃圾收集器(garbage collector)可以自動(dòng)地清理內(nèi)存。即使這樣,一切被拋棄的資源也需要適當(dāng)?shù)奶幚怼貏e是那些垃圾收集器不能管理的資源。

資源管理問(wèn)題的常見(jiàn)來(lái)源
內(nèi)存碎片 如果沒(méi)有足夠大的連續(xù)的虛擬地址存儲(chǔ)空間,可能會(huì)導(dǎo)致分配失敗
進(jìn)程限制 進(jìn)程通常都可以讀取內(nèi)存的所有子集,以及系統(tǒng)可用的資源。
資源泄露 垃圾收集器只管理內(nèi)存,其他資源需要由應(yīng)用程序正確管理。
不穩(wěn)定資源 那些依賴于垃圾收集器與終結(jié)器(finalizers)的資源在很久沒(méi)用過(guò)的時(shí)候,不可被立即調(diào)用。實(shí)際上它們可能永遠(yuǎn)不可能被調(diào)用。

利用try/finally block來(lái)確保資源已被合理釋放,或是讓你的類使用IDisposable,以及更方便更安全的聲明方式。

using (StreamReader reader=new StreamReader(file)) 
{ 
 //your code here

在產(chǎn)品代碼中避免garbage collector

除了用調(diào)用GC.Collect()干擾garbage collector之外,也可以考慮適當(dāng)?shù)蒯尫呕蚴菕仐壻Y源。在進(jìn)行性能測(cè)試時(shí),如果你可以承擔(dān)這種影響帶來(lái)的后果,你再去使用garbage collector。

避免編寫finalizers

與當(dāng)前一些流傳的謠言不同的是,你的類不需要Finalizers,而這只是因?yàn)镮Disposable的存在!你可以讓IDisposable賦予你的類在任何已擁有的組合實(shí)例中調(diào)用Dispose的能力,但是finalizers只能在擁有未管理的資源類中使用。

Finalizers主要對(duì)交互式Win32位句柄API有很大作用,并且SafeHandle句柄是很容易利用的。

不要總是設(shè)想你的finalizers(總是在finalizer線程上運(yùn)行的)會(huì)很好地與其他對(duì)象進(jìn)行交互。那些其他的對(duì)象可能在該進(jìn)程之前就被終止掉了。

12.并發(fā)性

處理并發(fā)性與多線程編程是件復(fù)雜的、困難的事情。在將并發(fā)性添加進(jìn)你的程序之前,請(qǐng)確保你已經(jīng)明確了解你的做的是什么——因?yàn)檫@里面有太多門道了!

多線程軟件的情況很難進(jìn)行預(yù)測(cè),比如很容易產(chǎn)生如競(jìng)爭(zhēng)條件與死鎖的問(wèn)題,而這些問(wèn)題并不是僅僅影響單線程應(yīng)用。基于這些風(fēng)險(xiǎn),你應(yīng)該將多線程視為最后一種手段。如果不得不使用多線程,盡量縮減多線程同時(shí)使用內(nèi)存的需求。如果必須使線程同步,請(qǐng)盡可能地使用最高等級(jí)的同步機(jī)制。在最高等級(jí)的前提下,包括了這些機(jī)制:

  • Async-await/Task Parallel Library/Lazy

  • Lock/monitor/AutoResetEvent

  • Interlocked/Semaphore

  • 可變域與顯式barrier

以上的這些很難解釋清楚C#/.NET的復(fù)雜之處。如果你想開(kāi)發(fā)一個(gè)正常的并發(fā)應(yīng)用,可以去參閱O’Reilly的《Concurrency in C# Cookboo》。

使用Volatile

將一個(gè)域標(biāo)記為“volatile”是一種高級(jí)特性,而這種設(shè)置也經(jīng)常被專家所誤解。C#的編譯器會(huì)保證目標(biāo)域可以被獲取與釋放語(yǔ)義,但是被lock的域就不適用于這種情況。如果你不知道獲取什么,不知道釋放什么語(yǔ)義,以及它們是怎樣影響CPU層次的優(yōu)化,那么久避免使用volatile域。取而代之的可以用更高層次的工具,比如Task Parallel Library或是CancellationToken。

線程安全與內(nèi)置方法

標(biāo)準(zhǔn)庫(kù)類型常提供使對(duì)象線程安全更容易的方法。例如Dictionary.TryGetValue()。使用此類方法一般可以使你的代碼變得更加清爽,并且你也不必?fù)?dān)心像TOCTOU(time-of-check-time-of-use競(jìng)爭(zhēng)危害的一種)這樣的數(shù)據(jù)競(jìng)爭(zhēng)。

不要鎖住“this”、字符串,或是其他普通public的對(duì)象

當(dāng)使用在多線程環(huán)境下的一些類時(shí),多注意lock的使用。鎖住字符串常量,或是其他公共對(duì)象,會(huì)阻止你鎖狀態(tài)下的封裝,還可能會(huì)導(dǎo)致死鎖。你需要阻止其他代碼鎖定在同一使用的對(duì)象上,當(dāng)然你最好的選擇是使用private對(duì)象成員項(xiàng)。

13.避免常見(jiàn)的錯(cuò)誤

Null

濫用null是一種常見(jiàn)的導(dǎo)致程序錯(cuò)誤的來(lái)源,這種非正常操作可能會(huì)使程序崩潰或是其他的異常。如果你試圖獲取一個(gè)null的引用,就好像它是某對(duì)象的有效引用值(例如通過(guò)獲取一個(gè)屬性或是方法),那么在運(yùn)行時(shí)就會(huì)拋出一個(gè)NullReferenceException。

靜態(tài)與動(dòng)態(tài)分析工具可以在你發(fā)布代碼之前為你檢查出潛在的NullReferenceException。在C#當(dāng)中,引用型為null通常是由于變量沒(méi)有引用到某個(gè)對(duì)象而造成的。對(duì)于值可為空的類型與引用型來(lái)說(shuō),是可以使用null的。例如:Nullable,空委托,已注銷的事件,“as”轉(zhuǎn)化失敗的,以及一些其他的情況。

每個(gè)null引用異常都是一個(gè)bug。相比于找到NullReferenceException這個(gè)問(wèn)題來(lái)說(shuō),不如嘗試在你使用該對(duì)象之前去為null進(jìn)行測(cè)試。這樣一來(lái)可以使代碼更易于最小化的try/catch block讀取。

當(dāng)從數(shù)據(jù)庫(kù)表中讀取數(shù)據(jù)時(shí),注意缺失值可以表示為DBNull 對(duì)象,而不是作為空引用。不要期望它們表現(xiàn)得像潛在的空引用一樣。

用二進(jìn)制的數(shù)字表示十進(jìn)制的值

Float與double都可以表示十進(jìn)制實(shí)數(shù),但不能表示二進(jìn)制實(shí)數(shù),并且在存儲(chǔ)十進(jìn)制值的時(shí)候可以在必要時(shí)用二進(jìn)制的近似值存儲(chǔ)。從十進(jìn)制的角度來(lái)看,這些二進(jìn)制的近似值通常都有不同的精度與取舍,有時(shí)在算數(shù)操作當(dāng)中會(huì)導(dǎo)致一些不期望的結(jié)果。由于浮點(diǎn)型運(yùn)算通常在硬件當(dāng)中執(zhí)行,因此硬件條件的不可預(yù)測(cè)會(huì)使這些差異更加復(fù)雜。

在十進(jìn)制精度很重要的時(shí)候,就要使用十進(jìn)制了——比如經(jīng)濟(jì)方面的計(jì)算。

調(diào)整結(jié)構(gòu)

有一種常見(jiàn)的錯(cuò)誤就是忘記了結(jié)構(gòu)是值類型,意即其復(fù)制與通過(guò)值傳遞。例如你可能見(jiàn)過(guò)這樣的代碼:

struct P { public int x; public int y; }
void M()
{
   P p = whatever;
   …
   p.x = something;
   …
   N(p);

忽然某一天,代碼維護(hù)人員決定將代碼重構(gòu)成這樣:

void M()
{
   P p = whatever;
   Helper(p);
   N(p);
}
void Helper(P p)
{ 
   …
   p.x = something;

現(xiàn)在當(dāng)N(p)在M()中被調(diào)用,p就有了一個(gè)錯(cuò)誤的值。調(diào)用Helper(p)傳遞p的副本,并不是引用p,于是在Helper()中的突變便丟失掉了。如果被正常調(diào)用,那么Helper應(yīng)該傳遞的是調(diào)整過(guò)的p的副本。

非預(yù)期計(jì)算

C#編譯器可以保護(hù)在運(yùn)算過(guò)程中的常量溢出,但不一定是計(jì)算值。使用“checked”與“unchecked”兩個(gè)關(guān)鍵詞來(lái)標(biāo)記你想對(duì)變量進(jìn)行什么操作。

不保存返回值

與結(jié)構(gòu)體不同的是,類是引用類型,并且可以適當(dāng)?shù)匦薷囊脤?duì)象。然而并不是所有的對(duì)象方法都可以實(shí)際修改引用對(duì)象,有一些返回的是一個(gè)新的對(duì)象。當(dāng)開(kāi)發(fā)者調(diào)用后者時(shí),他們需要記住將返回值分配給一個(gè)變量,這樣才可以使用修改過(guò)的對(duì)象。在代碼審查階段,這些問(wèn)題的類型通常會(huì)逃過(guò)審查而不被發(fā)現(xiàn)。像字符串之類的對(duì)象,它們是不可變的,因此永遠(yuǎn)不可能修改這些對(duì)象。即便如此,開(kāi)發(fā)者還是很容易忘記這些問(wèn)題。

例如,看如下 string.Replace()代碼:

string label = “My name is Aloysius”;
label.Replace(“Aloysius”, “secret”);

這兩行代碼運(yùn)行之后會(huì)打印出“My name is Aloysius” ,這是因?yàn)镽aeplace方法并沒(méi)改變?cè)撟址闹怠?/p>

不要使迭代器與枚舉器失效

注意不要在遍歷時(shí)去修改集合

List myItems = new List{20,25,9,14,50};
foreach(int item in myItems)
{
    if (item < 10)
    {
        myItems.Remove(item);
        // iterator is now invalid!
        // you’ll get an exception on the next iteration

如果你運(yùn)行了這個(gè)代碼,那么它一在下一項(xiàng)的集合中進(jìn)行循環(huán),你就會(huì)得到一個(gè)異常。

正確的處理方法是使用第二個(gè)list去保存你想刪除的這一項(xiàng),然后在你想刪除的時(shí)候再遍歷這個(gè)list:

List myItems = new List{20,25,9,14,50};
List toRemove = new List();
foreach(int item in myItems)
{
   if (item < 10)
   {
        toRemove.Add(item);         
   }
}
foreach(int item in toRemove)
{

如果你用的是C#3.0或更高版本,可以嘗試List.RemoveAll:

myInts.RemoveAll(item => (item < 10));

屬性名稱錯(cuò)誤

在實(shí)現(xiàn)屬性時(shí),要注意屬性的名稱和在類當(dāng)中用的成員項(xiàng)的名字有很大差別。很容易在不知情的情況下使用了相同的名稱,并且在屬性被獲取的時(shí)候還會(huì)觸發(fā)死循環(huán)。

// The following code will trigger infinite recursion
private string name;
public string Name
{
    get
    {
        return Name;  // should reference “name” instead.

在重命名間接屬性時(shí)同樣要小心。例如:在WPF中綁定的數(shù)據(jù)將屬性名稱指定為字符串。有時(shí)無(wú)意的改變屬性名稱,可能會(huì)不小心造成編譯器無(wú)法解決的問(wèn)題。

以上是“C#開(kāi)發(fā)者需要注意什么”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對(duì)大家有所幫助,如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注創(chuàng)新互聯(lián)成都網(wǎng)站設(shè)計(jì)公司行業(yè)資訊頻道!

另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無(wú)理由+7*72小時(shí)售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國(guó)服務(wù)器、虛擬主機(jī)、免備案服務(wù)器”等云主機(jī)租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡(jiǎn)單易用、服務(wù)可用性高、性價(jià)比高”等特點(diǎn)與優(yōu)勢(shì),專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場(chǎng)景需求。


分享文章:C#開(kāi)發(fā)者需要注意什么-創(chuàng)新互聯(lián)
本文鏈接:http://weahome.cn/article/gceoc.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部