C#入?yún)⑹褂靡妙愋鸵觬ef的原因解析
摘一段來自官網(wǎng)的說明 :方法的參數(shù)列表中使用 ref 關(guān)鍵字時(shí),它指示參數(shù)按引用傳遞,而非按值傳遞。 ref 關(guān)鍵字讓形參成為實(shí)參的別名,這必須是變量。 換而言之,對(duì)形參執(zhí)行的任何操作都是對(duì)實(shí)參執(zhí)行的。
大體意思就是將實(shí)慘的引用作為參數(shù)傳遞,如果入?yún)⒉患觬ef修飾,本身傳遞的是實(shí)參的值到方法中。
那什么是值,什么是引用?大體可以理解為堆棧的區(qū)別,在.net中大多數(shù)實(shí)例存在于托管堆棧中。struct,int32,int64,double,enum等數(shù)據(jù)類型聲明的實(shí)例存在棧中,而string,class,delegate等存在于堆中。前者一般稱為值類型,后者則叫做引用類型,那么引用類型和值類型在內(nèi)存中的大概圖如下:
可以看到引用類型被存儲(chǔ)的時(shí)候真正內(nèi)容存在堆中,而在stack中存儲(chǔ)了一個(gè)堆中地址的引用指向堆中真正內(nèi)容。
ref修飾入?yún)⒌某S脠?chǎng)景
當(dāng)我們希望一個(gè)已有返回值的方法能夠修改我一個(gè)外部基本類型的值的時(shí)候,我們可以將該參數(shù)加上ref關(guān)鍵字作為入?yún)?。具體原理其實(shí)就是將棧中具體的值替換為了棧的引用,說白了就是地址,幻想一下本來你高考作弊只是后臺(tái)改了一下自己成績(jī)和學(xué)霸一樣,學(xué)霸的人生沒有受到影響,現(xiàn)在你干脆把學(xué)霸的試卷改成了自己名字,學(xué)霸直接被你影響只能進(jìn)廠打螺絲了,當(dāng)然這個(gè)比喻不是很貼切,大家理解到意思就行。
那么為什么引用類型的入?yún)⑽覀儾恍枰砑觬ef?因?yàn)閺纳蠄D我們能看出來?xiàng)V写鎯?chǔ)的本來就是引用類型的地址,所以引用類型不需要添加ref關(guān)鍵字,當(dāng)你在方法內(nèi)部修改了入?yún)⒌囊恍傩灾?,外面的?shí)慘依舊會(huì)受到影響。
引用類型添加ref的作用是啥?
我們?cè)趯?shí)際開發(fā)中還是能夠碰到一些引用類型添加ref的場(chǎng)景。其實(shí)道理也是一樣的,就是將引用類型的棧的地址傳遞到了方法中,那么和不添加有啥區(qū)別?我們來看看下面的代碼:
Student student1 = new Student("Jack"); Student student2 = new Student("Lucy"); WithoutRef(student1); WithRef(ref student2); Console.WriteLine($"不帶ref的方法---{student1.Name}"); Console.WriteLine($"帶ref的方法---{student2.Name}"); Console.Read(); //不帶ref的方法 static void WithoutRef(Student stu) { stu.Name = "Bruce"; } //不帶ref的方法 static void WithRef(ref Student stu) { stu.Name = "Bruce"; } public class Student { public Student(string name) { Name = name; } public string Name { get; set; } }
可以看到結(jié)果一樣的,兩個(gè)實(shí)例的名字都變了
那我們?cè)倏纯聪旅娴拇a:
Student student1 = new Student("Jack"); Student student2 = new Student("Lucy"); WithoutRef(student1); WithRef(ref student2); Console.WriteLine($"不帶ref的方法---{student1.Name}"); Console.WriteLine($"帶ref的方法---{student2.Name}"); Console.Read(); //不帶ref的方法 static void WithoutRef(Student stu) { stu = new Student("Bruce"); } //不帶ref的方法 static void WithRef(ref Student stu) { stu = new Student("Bruce"); } public class Student { public Student(string name) { Name = name; } public string Name { get; set; } }
我們發(fā)現(xiàn)第一個(gè)不帶ref的方法名字沒有發(fā)生變化。
代碼的變化就是第二個(gè)例子重新new了一下,我們根據(jù)上面的原理分析就能知道為啥。由于ref將引用類型的棧地址傳遞過去了,new關(guān)鍵字大體就是在堆中開辟一個(gè)新空間,然后將空間地址存儲(chǔ)到棧中,由于ref將棧地址傳遞過來了,所以就將棧中的存儲(chǔ)地址替換為了新開辟的堆地址了,而不帶ref的引用類型本身傳遞的只是堆中地址的引用,所以new關(guān)鍵字等于說將形參重新開辟空間和分配了,和實(shí)參已經(jīng)不是同一個(gè)地方了。
總結(jié)
ref傳遞的是棧地址
不帶ref傳遞的是棧中存儲(chǔ)的值,可能是一個(gè)值,也可能是一個(gè)地址引用。
到此這篇關(guān)于C#入?yún)⑹褂靡妙愋鸵觬ef的原因解析的文章就介紹到這了,更多相關(guān)c#引用類型加ref內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C#實(shí)現(xiàn)把dgv里的數(shù)據(jù)完整的復(fù)制到一張內(nèi)存表的方法
這篇文章主要介紹了C#實(shí)現(xiàn)把dgv里的數(shù)據(jù)完整的復(fù)制到一張內(nèi)存表的方法,需要的朋友可以參考下2014-08-08C#數(shù)據(jù)結(jié)構(gòu)與算法揭秘一
本文一介紹了數(shù)據(jù)結(jié)構(gòu)的基本概念 而介紹了算法的基本概念,并且重點(diǎn)討論了算法時(shí)間復(fù)雜度,并且用程序予以證明2012-11-11在C#中如何使用正式表達(dá)式獲取匹配所需數(shù)據(jù)
本文給大家分享C#中如何使用正式表達(dá)式獲取匹配所需數(shù)據(jù) ,非常實(shí)用,對(duì)正則表達(dá)式獲取匹配相關(guān)知識(shí)感興趣的朋友一起學(xué)習(xí)吧2016-03-03WPF彈出右鍵菜單時(shí)判斷鼠標(biāo)是否選中該項(xiàng)
這篇文章介紹了WPF彈出右鍵菜單時(shí)判斷鼠標(biāo)是否選中該項(xiàng)的方法,文中通過示例代碼介紹的非常詳細(xì)。對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-06-06