自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

C++函數(shù)返回指針和引用的坑

開發(fā) 前端
今天這篇文章我們來看看在C++新手們針對(duì)指針和引用的使用經(jīng)常犯的錯(cuò)誤。

常用C++進(jìn)行項(xiàng)目開發(fā)的童鞋們應(yīng)該都知道,在C++中指針和引用是常用的語法了,而指針又是C++區(qū)別于其他高級(jí)語言的一大精髓。

而今天我們?cè)賮砜纯丛贑++新手們針對(duì)指針和引用的使用經(jīng)常犯的錯(cuò)誤。

函數(shù)返回指針

在C++中針對(duì)一個(gè)函數(shù)返回指針的實(shí)現(xiàn)方式一般有三種:

1.返回一個(gè)變量的地址

例如以下代碼:

// 返回int指針地址
int * funTest(){
    int a = 101;
    return &a;
}

int main(int argc, const char *argv[]) {
    int *a = funTest();
    std::cout << "a的值:" << *a << std::endl;
    return 0;
}

以上代碼在筆者的電腦上運(yùn)行就直接報(bào)錯(cuò)崩潰了,崩潰信息:

Process finished with exit code 139 (interrupted by signal 11: SIGSEGV)

上面的代碼返回一個(gè)局部變量a的地址,這個(gè)變量a緊在函數(shù)funTest內(nèi)有效,當(dāng)函數(shù)funTest結(jié)束了,變量a的生命周期也就結(jié)束了, 此時(shí)變量a所占用的內(nèi)存空間將被釋放,因此返回的指針地址將會(huì)被指向一個(gè)未知數(shù),后續(xù)再使用這個(gè)指針是未定義的行為,可能會(huì)導(dǎo)致程序崩潰或者出現(xiàn)其他異常。

針對(duì)這樣的危險(xiǎn)代碼行為,其實(shí)編輯器也已經(jīng)給出了警告,所以說在開發(fā)過程中也不要以為的忽略警告哦。

為了杜絕此類行為的發(fā)生,還可以使用筆者之前的介紹的代碼質(zhì)量檢測(cè)工具cppcheck進(jìn)行檢測(cè),在開發(fā)過程中直接劃線提醒。

介紹一款CPP代碼bug檢測(cè)神器。

2.返回一個(gè)使用static修飾的變量地址

我們修改一下funTest函數(shù)的變量a,使用static關(guān)鍵字修飾一下:

// 返回int指針地址
int * funTest(){
    static int a = 101;
    return &a;
}

int main(int argc, const char *argv[]) {
    int *a = funTest();
    std::cout << "a的值:" << *a << std::endl;
    return 0;
}

運(yùn)行發(fā)現(xiàn)程序并沒有崩潰,而且是正確打印出了變量a的值。這是因?yàn)?使用static 表示將這個(gè)變量存儲(chǔ)到全局區(qū)(static靜態(tài)區(qū)), 此時(shí)就不受棧區(qū)管控,當(dāng)函數(shù)funTest執(zhí)行完畢后,變量a依然存在,不會(huì)存在前面所說的變量地址被釋放的問題。

3.使用動(dòng)態(tài)分配內(nèi)存new關(guān)鍵字

int * funTest(){
    //動(dòng)態(tài)分配的內(nèi)存空間,手動(dòng)delete后才會(huì)釋放
    int* a = new int(101) ;
    return a;
}

int main(int argc, const char *argv[]) {
    int *a = funTest();
    std::cout << "a的值:" << *a << std::endl;
    return 0;
}

上述代碼不會(huì)崩潰,也能正常運(yùn)行,但是存在一個(gè)隱患就是返回的指針變量a如果忘記調(diào)用delete則會(huì)造成內(nèi)存泄露, 這就引發(fā)了一個(gè)指針變量誰維護(hù)銷毀的問題。一般默認(rèn)規(guī)則是誰開發(fā)維護(hù)。

因此,針對(duì)這樣的場(chǎng)景,筆者的建議是智能指針你值得擁有...

函數(shù)返回一個(gè)引用

我們看看以下返回一個(gè)引用的例子代碼:

int & funTest(){
    //動(dòng)態(tài)分配的內(nèi)存空間,手動(dòng)delete后才會(huì)釋放
    int a = 101 ;
    return a;
}

int main(int argc, const char *argv[]) {
    int a = funTest();
    std::cout << "a的值:" << a << std::endl;
    return 0;
}

筆者在CLion上測(cè)試也是直接崩潰了,原因也是和上面所說的返回一個(gè)局部變量的地址一樣, 都是因?yàn)楹瘮?shù)funTest結(jié)束后,變量a的生命周期結(jié)束了, 變量a也就是被釋放了,再返回它的引用的話就是未定義的。至于為什么它們的原因是一樣的呢?因?yàn)樗^引用,可以簡(jiǎn)單地理解為引用其實(shí)就是帶const修飾的指針。

那么針對(duì)這個(gè)問題該如何修正呢?首先使用static關(guān)鍵字肯定是可以的。那么使用動(dòng)態(tài)內(nèi)存new的方式行不行呢?答案也是可行的,但是需要注意的一點(diǎn)就是如果一個(gè)引用 的值來源于一個(gè)指針,后來這個(gè)指針被delete掉了,那么再使用這個(gè)引用也是會(huì)造成崩潰的...

如何返回一個(gè)數(shù)組

那么問題來了,舉一反三,如果想通過一個(gè)函數(shù)返回一個(gè)數(shù)組那該如何實(shí)現(xiàn)呢?

眾所周知,C++是不允許直接返回一個(gè)數(shù)組的,如果您想要從函數(shù)返回一個(gè)一維數(shù)組,您必須聲明一個(gè)返回指針的函數(shù)。

例如下面的寫法是編譯不通過的:

// 無法編譯通過,不能返回一個(gè)數(shù)組
int[] funTest(){
    int myArray[3] = {1, 2, 3};
    return myArray;
}

正確的寫法應(yīng)該是:

int* funTest(){
    static int myArray[3] = {1, 2, 3};
    return myArray;
}

因而可以看出,其實(shí)返回一個(gè)數(shù)組的函數(shù)所遇到的坑其實(shí)就轉(zhuǎn)換成了返回一個(gè)指針的函數(shù)所遇到的坑,這些坑的舉例就如前面所說...

責(zé)任編輯:趙寧寧 來源: 思想覺悟
相關(guān)推薦

2011-07-13 16:14:53

C++引用指針

2010-01-18 15:53:27

C++析構(gòu)函數(shù)

2014-01-24 09:49:01

C++指針

2015-05-13 10:37:58

C++指針與引用

2021-07-28 06:53:02

C++Const指針傳遞

2024-05-15 16:01:04

C++編程開發(fā)

2010-01-11 15:29:13

引用C++語言

2024-02-22 14:06:39

C++指針開發(fā)

2011-07-15 01:20:58

C指針函數(shù)函數(shù)指針

2022-07-26 00:36:06

C#C++函數(shù)

2021-06-10 08:51:57

C++指針聲明指針相關(guān)概念

2011-04-11 11:09:50

this指針

2010-02-06 09:31:42

C++函數(shù)對(duì)象

2011-07-20 17:54:02

C++

2021-06-18 12:30:36

C++函數(shù)指針編程語言

2021-12-21 15:31:10

C++語言指針

2010-01-25 10:10:42

C++函數(shù)參數(shù)

2010-02-02 15:01:59

C++成員函數(shù)指針

2010-02-01 09:18:49

C++函數(shù)指針

2011-07-20 16:09:08

C++
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)