C#類型轉(zhuǎn)化的五點(diǎn)心得
對(duì)于這種C#類型轉(zhuǎn)化我們通常用兩種選擇:使用as操作符,或者使用強(qiáng)制轉(zhuǎn)型。當(dāng)然還有一個(gè)做法是先用is測(cè)試轉(zhuǎn)換是否可行,然后再用as或者強(qiáng)制轉(zhuǎn)換。
本文主要給大家提醒一些使用C#類型轉(zhuǎn)化時(shí)需要注意的地方,或許你更想弄清楚as和強(qiáng)制轉(zhuǎn)換的區(qū)別和使用場(chǎng)景,不用著急,看完,本文你自己就知道答案了,那么上面提到的兩種C#類型轉(zhuǎn)化我們需要注意哪些呢?
1、as和is操作符都不執(zhí)行任何用戶自定義的轉(zhuǎn)換。
2、對(duì)于強(qiáng)制轉(zhuǎn)換,引用為空將會(huì)轉(zhuǎn)換出錯(cuò)。
3、強(qiáng)制轉(zhuǎn)換任意類型和自定義轉(zhuǎn)換兩種情況的IL代碼展示有區(qū)別。
4、用戶自定義轉(zhuǎn)換只作用于對(duì)象的編譯時(shí)類型。
5、as操作符不能應(yīng)用于值類型。
6、foreach循環(huán)語(yǔ)句中使用強(qiáng)制類型轉(zhuǎn)型。
下面我們逐一進(jìn)行介紹說(shuō)明:
1,我們先來(lái)看錯(cuò)誤代碼示例:
- classA
- {
- }
- classC
- {
- publicstaticimplicitoperatorA(Ct)
- {
- returnnewA();
- }
- }
- classProgram
- {
- staticvoidMain(string[]args)
- {
- objecto=Factory.GetObject();
- //o為一個(gè)C類型:
- Aa=oasA;//轉(zhuǎn)型失敗,o的類型不是A
- }
- }
代碼已經(jīng)很明顯,我們不可以因?yàn)槎x了C到A的強(qiáng)制轉(zhuǎn)換,就使用as,對(duì)應(yīng)用戶自定義轉(zhuǎn)換,我們只可以使用(A)o轉(zhuǎn)換,其實(shí)自定義轉(zhuǎn)換和自定義操作=、+、-、"、%等操作符是一樣的機(jī)理,這樣你就明白為什么用as不可以使用自定義轉(zhuǎn)換了。
2,還是看一個(gè)錯(cuò)誤示例:
- classA
- {
- }
- classB:A
- {
- }
- classProgram
- {
- staticvoidMain(string[]args)
- {
- Bb;
- Aa=(A)b;
- }
- }
- 問(wèn)題不用多言,as可以解決這個(gè)問(wèn)題。
- 3,看一個(gè)示例:
- classA
- {
- }
- classC
- {
- publicstaticimplicitoperatorA(Ct)
- {
- returnnewA();
- }
- }
- classB:A
- {
- }
- classProgram
- {
- staticvoidMain(string[]args)
- {
- Aa=newB();
- Bb=(B)a;
- Cc=newC();
- a=(A)c;
- }
- }
- 對(duì)于Bb=(B)a;的IL代碼如下:
- IL_0008:castclassConsoleApplication1.B
- 對(duì)于a=(A)c;的IL代碼如下:
- IL_0015:callclassConsoleApplication1.AConsoleApplication1.C::op_Implicit(classConsoleApplication1.C)
- 區(qū)別大家已經(jīng)看到了,要說(shuō)真正認(rèn)識(shí)區(qū)別那么我們要繼續(xù)談“4、用戶自定義轉(zhuǎn)換只作用于對(duì)象的編譯時(shí)類型。"
- 4,看一個(gè)示例:
- classA
- {
- }
- classC
- {
- publicstaticimplicitoperatorA(Ct)
- {
- returnnewA();
- }
- }
- classB:A
- {
- }
- classProgram
- {
- staticvoidMain(string[]args)
- {
- Aa=newB();
- Bb=(B)a;
- objectc=newC();
- a=(A)c;//編譯通過(guò),運(yùn)行失??!
- }
- }
也許你從來(lái)沒(méi)想過(guò)a=(A)c;編譯會(huì)成功,運(yùn)行會(huì)出錯(cuò)在這句,畢竟我們有一個(gè)顯示類型C到類型A的轉(zhuǎn)換操作,我們看看IL代碼即可找到答案,
- .methodprivatehidebysigstaticvoidMain(string[]args)cilmanaged
- {
- .entrypoint
- //Codesize28(0x1c)
- .maxstack1
- .localsinit([0]classConsoleApplication1.Aa,
- [1]classConsoleApplication1.Bb,
- [2]objectc)
- IL_0000:nop
- IL_0001:newobjinstancevoidConsoleApplication1.B::.ctor()
- IL_0006:stloc.0
- IL_0007:ldloc.0
- IL_0008:castclassConsoleApplication1.B
- IL_000d:stloc.1
- IL_000e:newobjinstancevoidConsoleApplication1.C::.ctor()
- IL_0013:stloc.2
- IL_0014:ldloc.2
- IL_0015:castclassConsoleApplication1.A
- IL_001a:stloc.0
- IL_001b:ret
- }//endofmethodProgram::Main
大家注意看IL_0015:castclassConsoleApplication1.A這句,這句說(shuō)明自定義轉(zhuǎn)換在編譯時(shí)刻進(jìn)行,也許你要問(wèn)為什么不是第3條的:
IL_0015:callclassConsoleApplication1.AConsoleApplication1.C::op_Implicit(classConsoleApplication1.C)
因?yàn)閛bjectc,c被定義為Object類型,那么強(qiáng)制轉(zhuǎn)換在編譯時(shí)刻去Object找是否存在自定義轉(zhuǎn)換操作(注意,()轉(zhuǎn)型時(shí)編譯器優(yōu)先考慮自定義轉(zhuǎn)換,找不到才進(jìn)行castclass),當(dāng)然Object沒(méi)有自定義轉(zhuǎn)換為A的操作,那么就使用普通的強(qiáng)制轉(zhuǎn)換castclass。好了現(xiàn)在我們知道了用戶自定義轉(zhuǎn)換只作用于對(duì)象的編譯時(shí)類型,而普通的Bb=(B)a;強(qiáng)制轉(zhuǎn)換可以作用到運(yùn)行時(shí)刻。那么上面的錯(cuò)誤如何去掉呢?對(duì)應(yīng)代碼修改為:
- objectc=newC();
- Ccc=casC;
- a=(A)cc;
現(xiàn)在運(yùn)行正常通過(guò),好了我們?cè)俅尾榭碔L代碼
IL_001c:callclassConsoleApplication1.AConsoleApplication1.C::op_Implicit(classConsoleApplication1.C)
到這大家對(duì)3和4點(diǎn)的認(rèn)識(shí)應(yīng)該很清楚了吧。
5、as操作符不能應(yīng)用于值類型————省略!,鑒于這個(gè)點(diǎn)很簡(jiǎn)單,本人就不提供示例了,大家有興趣可以自己試驗(yàn)下。
6、看代碼示例:(以下代碼摘自《EffectiveC#中文版改善C#程序的50中方法》——23頁(yè))
- publicvoidUseCollection(IEnumerabletheCollection)
- {
- foreach(MyTypetintheCollection)
- t.DoStuff();
- }
- //上面代碼等同于:
- publicvoidUseCollection(IEnumerabletheCollection)
- {
- IEnumeratorit=hteCollection.GetEnumerator();
- while(it.MoveNext())
- {
- MyTypet=(MyType)it.Current;
- t.DoStuff();
- }
- }
通過(guò)查看IL代碼我們可以確認(rèn)foreach語(yǔ)句的轉(zhuǎn)換是使用的強(qiáng)制轉(zhuǎn)換操作,那么為什么呢?之所以使用強(qiáng)制轉(zhuǎn)型,是因?yàn)閒oreach語(yǔ)句需要同時(shí)支持值類型和引用類型,這側(cè)面說(shuō)明我們的第5點(diǎn)as不支持值類型。
好了,5點(diǎn)C#類型轉(zhuǎn)化說(shuō)明已經(jīng)解釋完了,你現(xiàn)在還想問(wèn)as和強(qiáng)制轉(zhuǎn)換的區(qū)別和使用場(chǎng)景么?,歡迎提出批評(píng)、指正錯(cuò)誤。
【編輯推薦】