鍍金池/ 教程/ C/ 解析Console.WriteLine()
解析接口
C#中一些易混淆概念總結(jié)-數(shù)據(jù)類型存儲(chǔ)位置,方法調(diào)用,out和ref參數(shù)的使用
解析里氏替換原則,虛方法
繼承
構(gòu)造函數(shù),this關(guān)鍵字,部分類,枚舉
解析Console.WriteLine()
解析抽象類,抽象方法
結(jié)構(gòu),GC回收,靜態(tài)成員,靜態(tài)類

解析Console.WriteLine()

這幾天在溫習(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)和類的了解有沒有更深入的了解呢?