關於C#函數對象參數傳遞的問題


轉自:   https://www.cnblogs.com/qguohog/archive/2009/12/26/1632967.html

 

我們知道在.net里class是引用類型,在函數參數表中的對象傳遞的都是對象的引用,所以在函數體內對其對象參數的修改會影響函數外對應的對象本身,例如下面的程序:

 

    class Program
    {
        static void Main(string[] args)
        {
            TestClass objA = new TestClass();
            objA.Name = "I am ObjA";

            Console.WriteLine(String.Format("In Main:{0}", objA.Name));

            TestFun(objA);

            Console.WriteLine(String.Format("In Main:{0}", objA.Name));

            Console.Read();
        }

        static void TestFun(TestClass obj)
        {
            obj.Name = "I am be modified in TestFun";
            Console.WriteLine(String.Format("In TestFun:{0}", obj.Name));
        }

        public class TestClass
        {
            public string Name { get; set; }
        }
    }

  將對象objA作為參數傳遞給函數TestFun,這時傳遞的是對象objA的引用,所以在函數內對objA的修改會直接應用到這個對象本身。在函數TestFun內修改了形參obj的Name屬性,所以在Main中再調用objA.Name時這個屬性值就變為在TestFun中修改后的值了。這一點很好理解,運行結果也就是:

image

 

 

    如果我們換一種寫法:

    class Program
    {
        static void Main(string[] args)
        {
            TestClass objA = new TestClass();
            objA.Name = "I am ObjA";

            Console.WriteLine(String.Format("In Main:{0}", objA.Name));

            TestFun(objA);

            Console.WriteLine(String.Format("In Main:{0}", objA.Name));

            Console.Read();
        }

        static void TestFun(TestClass obj)
        {
            TestClass objB = new TestClass();
            obj = objB;
            obj.Name = "I am ObjB";
            Console.WriteLine(String.Format("In TestFun:{0}", obj.Name));
        }

        public class TestClass
        {
            public string Name { get; set; }
        }
    }

    那結果又將如何呢? 在TestFun中新創建了一個對象objB,接着將obj指向新創建的對象objB並對其的Name屬性賦值,所以在TestFun函數體里Console.WriteLine的時候obj.Name應該是“I am ObjB”。那在Main里調用了TestFun(objA)后objA指向的對象到底做了改變沒有,它仍舊指向原來的對象還是已經指向了在TestFun里創建的objB呢?

    我的猜測是這樣的,將objA傳遞給TestFun傳遞的實際是objA的地址,在調用TestFun之前的對應關系是:

image

   objA是個變量,他本身的地址是000001,它存放的是objA對象的地址00E001。當調用函數TestFun(objA)時,由於要將objA作為參數傳遞給TestFun,所以在
棧中就復制一個objA地址的副本,這個副本的地址為000003,他存放的仍舊是objA對象的地址00E001,如圖:

image

   所以在TestFun函數中,一開始obj指向的還是objA對象。當在TestFun中創建了一個對象objB后,內存模型變為:

  image

  TestFun中執行“obj=objB”后,obj指向objB對象,如下圖:

image

 

 

 

 

  所以執行完TestFun函數后,Main函數中objA仍舊指向原來的objA對象,故運行結果如下:

image

 

本文中的內存模型僅是個人猜測,肯定有不正確或者不嚴謹的地方,希望不要誤導大家,也歡迎指正。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM