這幾天在溫習(xí)結(jié)構(gòu)體和類的時(shí)候遇到一個(gè)問題。發(fā)現(xiàn)一個(gè)奇怪的現(xiàn)象,一直找不到合理的答案。但是今天終于找到了合理的答案,所以拿來和大家分享一下。
class Program
{
static void Main(string[] args)
{
Point p;
Console.WriteLine(p);
Point p1 = new Point();
Console.WriteLine(p1);
Console.ReadKey();
}
}
//定義結(jié)構(gòu)
struct Point
{
////定義時(shí)賦初始值,編譯器會(huì)報(bào)錯(cuò)
//private int x;
//public Point()
//{ }
}
class Person
{
//在類中我們可以為屬性賦初始值
//private int nAge = 5;
//public int NAge
//{
// get { return nAge; }
// set { nAge = value; }
//}
}
當(dāng)我們只是聲明一個(gè)類和一個(gè)結(jié)構(gòu)體的時(shí)候,我們的編譯器順利的編譯通過。并且打印出結(jié)果如下:
http://wiki.jikexueyuan.com/project/csharp-confusing-concepts-summary/images/4.1.png" alt="" />
為什么我們沒有在結(jié)構(gòu)和類中做任何操作,卻可以打印出結(jié)果,且是"命名空間+"."+數(shù)據(jù)類型"呢?
首先我查閱了MSDN的關(guān)于結(jié)構(gòu)(struct)的官方文檔(地址點(diǎn)擊這里),有如下的一段話:
http://wiki.jikexueyuan.com/project/csharp-confusing-concepts-summary/images/4.10.png" alt="" />
結(jié)構(gòu)默認(rèn)的構(gòu)造函數(shù)(如果沒有顯式聲明)在實(shí)例化的時(shí)候才會(huì)被調(diào)用。所以,
//結(jié)構(gòu)的實(shí)例化可以不使用NEW關(guān)鍵字,只是將p加載到??臻g中,但是對(duì)象不可用,這里沒有調(diào)用默認(rèn)的構(gòu)造函數(shù)
Point p2;
Console.WriteLine(p2);
Console.WriteLine(p2);
Console.ReadKey();
在內(nèi)存中是如下的情況:
http://wiki.jikexueyuan.com/project/csharp-confusing-concepts-summary/images/4.2.png" alt="" />
此時(shí)在棧中已經(jīng)存在了p這個(gè)對(duì)象,但是不可用。
那么為什么會(huì)打印出"命名空間+"."+數(shù)據(jù)類型"的結(jié)果呢?
我們先看一下VS編譯后的中間代碼,即Msil,詳細(xì)解釋在圖中給出:
http://wiki.jikexueyuan.com/project/csharp-confusing-concepts-summary/images/4.3.png" alt="" />
有中間語言代碼,我們可以知道,最后調(diào)用的是Console.WriteLine(Object)方法
這時(shí)候就要深入的研究一下Console類了,用反編譯工具.NET Reflector查看Console類,因?yàn)樵谏厦娴拇a中,傳進(jìn).WriteLine()方法的是一個(gè)類,所以,我們要查看它的的(object value)方法,如下圖:
http://wiki.jikexueyuan.com/project/csharp-confusing-concepts-summary/images/4.4.png" alt="" />
這時(shí)候,我們?cè)偕钊氲絎riteLine()方法中去,源代碼,如下:
http://wiki.jikexueyuan.com/project/csharp-confusing-concepts-summary/images/4.5.png" alt="" />
再看Out.WriteLine()的源代碼:
http://wiki.jikexueyuan.com/project/csharp-confusing-concepts-summary/images/4.6.png" alt="" />
因?yàn)閜已經(jīng)在棧中創(chuàng)建了對(duì)象(但是不可用),所以,直接進(jìn)入else語句。
明顯的可以發(fā)現(xiàn)IFormattable是一個(gè)接口,我們?cè)倏碔Formattable接口的源碼,如下:
http://wiki.jikexueyuan.com/project/csharp-confusing-concepts-summary/images/4.7.png" alt="" />
顯然我們的Point 結(jié)構(gòu)沒有實(shí)現(xiàn)一個(gè)ToString()方法,不存在繼承關(guān)系,所以會(huì)轉(zhuǎn)化失敗,返回一個(gè)null值,又進(jìn)入下一個(gè)else語句
else
{
this.WriteLine(value.ToString());
}
這時(shí)候最重要的就要來了,我們看到value值被轉(zhuǎn)換為字符串輸出了,在看ToString()源代碼,如下:
http://wiki.jikexueyuan.com/project/csharp-confusing-concepts-summary/images/4.8.png" alt="" />
很明顯的發(fā)現(xiàn),是獲取該對(duì)象的數(shù)據(jù)類型并且轉(zhuǎn)化為字符串輸出。如下代碼:
Point p;
//打印出p的數(shù)據(jù)類型
Console.WriteLine(p.GetType());
Point p2;
Console.WriteLine(p2);
//使用NEW實(shí)例化了對(duì)像,調(diào)用了默認(rèn)的構(gòu)造函數(shù)
Point p1 = new Point();
Console.WriteLine(p1);
Console.ReadKey();
打印結(jié)果:
http://wiki.jikexueyuan.com/project/csharp-confusing-concepts-summary/images/4.9.png" alt="" />
這樣對(duì)結(jié)構(gòu)和類的了解有沒有更深入的了解呢?