5、應(yīng)用
我們知道實(shí)參的改變不影響形參,所以這種寫法并不能改變值,因?yàn)榇丝淌?傳值調(diào)用 :
按照之前 c 的寫法,我們使用 傳址調(diào)用 ,用指針修改:
但是學(xué)習(xí)引用之后,完全可以用引用修改:
x 和 y 分別是 a 和 b 的引用,對 x 和 y 進(jìn)行修改,就是對 a 和 b 進(jìn)行修改,所以值也被修改成功了。
調(diào)試一下:
它們的地址是完全相同的。而這里這里既不是傳值調(diào)用,也不是傳址調(diào)用,而是傳引用調(diào)用。
思考:上面三個(gè)函數(shù)是否構(gòu)成函數(shù)重載?構(gòu)成,但無法調(diào)用。
根據(jù)函數(shù)名修飾規(guī)則,傳值和傳引用的是不一樣的,比如會(huì)加上 R 做區(qū)分。
但是不能同時(shí)調(diào)用傳值和傳引用,因?yàn)橛衅缌x,就會(huì)導(dǎo)致調(diào)用不明確 ,編譯器并不知道調(diào)用哪個(gè):
引用解決二級指針生澀難懂的問題 :
講單鏈表時(shí),我們寫的由于是沒有頭結(jié)點(diǎn)的鏈表,所以修改時(shí),需要二級指針,對于指針概念不清晰的小伙伴們可能比較難理解。
但是學(xué)了引用,就可以解決這個(gè)問題:
結(jié)構(gòu)定義:
typedef structSListNode{
int data;
struct SListNode* next;
}SLTNode;
原代碼:
void SListPushFront(SLTNode** pphead, SLTDateType x)
{
SLTNode* newnode = BuyListNode(x);
newnode->next = *pphead;
*pphead = newnode;
}
// 調(diào)用
SLTNode* pilst = NULL;
SListPushFront(&plist);
修改后:
void SListPushFront(SLTNode*& pphead, SLTDateType x) // 改
{
SLTNode* newnode = BuyListNode(x);
newnode->next = *pphead;
*pphead = newnode;
}
// 調(diào)用
SLTNode* pilst = NULL;
SListPushFront(plist); // 改
修改之后的代碼里的二級指針被替換成了引用。
而這里的意思就是給一級指針取了一個(gè)別名,傳過來的是plist,而plist 是一個(gè)一級指針,所以會(huì)出現(xiàn) * ,而這里就相當(dāng)于 pphead 是 plist 的別名。而這里修改 pphead ,也就可以對 plist 完成修改。
但是有時(shí)候也會(huì)這么寫 :
結(jié)構(gòu)改造:
typedef structSListNode
{
int data;
struct SListNode* next;
}SLTNode, *PSLTNode;
這里的意思就是將 struct SListNode*
類型重命名為 PSLTNode
。
代碼:
void SListPushFront(PSLTNode& pphead, SLTDateType x) // 改
{
PSLTNode newnode = BuyListNode(x);
newnode->next = pphead;
pphead = newnode;
}
// 調(diào)用
PSLTNode plist = NULL;
SListPushFront(plist);
在 typedef
之后,PSLTNode
就是結(jié)構(gòu)體指針,所以傳參過去,只需要在形參那邊用引用接收,隨后進(jìn)行操作,就可以達(dá)成目的。
而形參的改變影響實(shí)參的參數(shù)叫做輸出型參數(shù),對于輸出型參數(shù),使用引用就十分舒適。
如果了解引用,那么這一部分是相當(dāng)好理解的,一些數(shù)據(jù)結(jié)構(gòu)教科書上也是這么寫的,但是如果不懂引用,甚至?xí)X得比二級指針還難以理解。
在我們學(xué)習(xí)了引用之后,之后也可以這么寫代碼,更加方便。
- 做返回值
要搞清楚這一塊,我們先進(jìn)行些許鋪墊。
int add(int a, int b)
{
int c = a + b;
return c;
}
int main()
{
int ret = add(1, 2);
cout << ret << endl;
return 0;
}
這里看似很簡單,就是把a(bǔ)dd函數(shù)計(jì)算結(jié)束的結(jié)果返回,但是這里包含了 傳值返回 。
若從棧幀角度看,會(huì)先創(chuàng)建 main 函數(shù)的棧幀,里面就會(huì)有 call 指令,開始調(diào)用 add 函數(shù)。而 add 函數(shù)也會(huì)形成棧幀,而棧幀中也有兩塊小空間,用來接受參數(shù),分別為 a 和 b,而里面的 c 則用來計(jì)算結(jié)果并返回。
而對于傳值返回,返回的并不是 c ,而是返回的是 c 的拷貝。而這其中會(huì)有一個(gè)臨時(shí)變量,返回的是臨時(shí)變量(見函數(shù)棧幀)
如果返回的是 c 的話,由于 add 的函數(shù)棧幀已經(jīng)銷毀了,就會(huì)產(chǎn)生很多奇怪的問題。c 能不能取到都是未知,而這時(shí)都是非法訪問,因?yàn)榭臻g已經(jīng)被歸還給系統(tǒng)了,所以必定是c拷貝后的數(shù)據(jù)被返回。
-
C語言
+關(guān)注
關(guān)注
180文章
7604瀏覽量
136824 -
C++
+關(guān)注
關(guān)注
22文章
2108瀏覽量
73651 -
面向?qū)ο?/span>
+關(guān)注
關(guān)注
0文章
64瀏覽量
9985
發(fā)布評論請先 登錄
相關(guān)推薦
評論