您好,登錄后才能下訂單哦!
這篇文章主要介紹“C++中 ‘=default ’及‘ =delete ’如何使用”,在日常操作中,相信很多人在C++中 ‘=default ’及‘ =delete ’如何使用問(wèn)題上存在疑惑,小編查閱了各式資料,整理出簡(jiǎn)單好用的操作方法,希望對(duì)大家解答”C++中 ‘=default ’及‘ =delete ’如何使用”的疑惑有所幫助!接下來(lái),請(qǐng)跟著小編一起來(lái)學(xué)習(xí)吧!
前言:
C++的類(lèi)有四類(lèi)特殊成員函數(shù),它們分別是:默認(rèn)構(gòu)造函數(shù)、析構(gòu)函數(shù)、拷貝構(gòu)造函數(shù)、拷貝賦值運(yùn)算符。如果實(shí)際編碼時(shí)沒(méi)有顯示定義,那么編譯器將會(huì)默認(rèn)生成這四類(lèi)成員函數(shù)。使用=default
和=delete
可以控制編譯器默認(rèn)函數(shù)體的使用。
C++11
新增了=default
標(biāo)識(shí),編譯器看到后,會(huì)生成默認(rèn)的執(zhí)行效率更高的函數(shù)定義體,同時(shí)會(huì)減輕編碼時(shí)的工作量。當(dāng)然,這里會(huì)引入一個(gè)問(wèn)題,既然編譯器會(huì)默認(rèn)生成構(gòu)造函數(shù),那么=default
的優(yōu)勢(shì)在哪里呢?回答這個(gè)問(wèn)題之前,先看下這段代碼
class Test{ public: Test(int a):x(a){}; private: int x; }; int main () { Test test; return 0; }
眾所周知,上面這段是編譯不過(guò)的,原因是因?yàn)樵赥est類(lèi)中我們自己定義了一個(gè)構(gòu)造函數(shù),編譯器看到后就不會(huì)再生成默認(rèn)構(gòu)造函數(shù)給我們,如果要解決這個(gè)編譯問(wèn)題的話需要我們提供一個(gè)沒(méi)有參數(shù)的構(gòu)造函數(shù)。
如:
Test(){};
在類(lèi)中加了上面的代碼之后,編譯器就會(huì)編譯通過(guò),但是在試想一下,如果這個(gè)類(lèi)很大,且需要我們?cè)陬?lèi)中初始化的成員很多呢?這個(gè)時(shí)候我們需要提供的這個(gè)默認(rèn)構(gòu)造函數(shù)就變得的很龐大,浪費(fèi)我們很多時(shí)間進(jìn)行變量的初始化,寫(xiě)一堆沒(méi)有技術(shù)的賦值或者其它初始化語(yǔ)句。同樣,拷貝構(gòu)造函數(shù)和拷貝賦值函數(shù)也是一樣。
=default
就給我們提供了這樣一個(gè)功能,加上之后,編譯器就會(huì)給我們默認(rèn)生成函數(shù)體,減輕工作量。
上面的類(lèi)就這可以這些寫(xiě):
class Test{ public: Test(int a):x(a){}; Test()=default; private: int x; };
當(dāng)然,=default
不但可以在類(lèi)成員內(nèi)部添加也可以在類(lèi)之外添加,但是使用=default
時(shí),必須遵守一個(gè)準(zhǔn)則:default 函數(shù)特性只能用于類(lèi)的特殊成員函數(shù)或者函數(shù)沒(méi)有默認(rèn)參數(shù)。 =default
寫(xiě)在類(lèi)之外的方式如下:
class Test{ public: Test(int a):x(a){}; Test()=default; Test(const Test& ts); Test& operator = (const Test& ts); private: int x; }; Test::Test(const Test& ts) =default; Test& Test::operator = (const Test& ts) =default;
上面的代碼中演示了=default在類(lèi)成員外部使用的場(chǎng)景。但是類(lèi)中確沒(méi)有析構(gòu)函數(shù),編碼時(shí),如果涉及到類(lèi)的繼承和派生,尤其是通過(guò)基類(lèi)指針指向了派生類(lèi)對(duì)象,當(dāng)調(diào)用delete刪除派生對(duì)象時(shí),如果基類(lèi)沒(méi)有顯示定義析構(gòu)函數(shù),編譯器會(huì)為基類(lèi)默認(rèn)生成析構(gòu)函數(shù),基類(lèi)對(duì)象會(huì)被正常釋放,但是也會(huì)產(chǎn)生一個(gè)問(wèn)題,派生類(lèi)沒(méi)有正確釋放,可能會(huì)產(chǎn)生內(nèi)存泄露等問(wèn)題。正確解決這種問(wèn)題的做法是在基類(lèi)中顯示定義一個(gè)虛析構(gòu)函數(shù)。這種方法在C++11之前是我們解決這種問(wèn)題經(jīng)常使用的,但是C++11之后,可以使用=default
,從而減輕我們的編碼量,且編譯器生成的代碼效率更高。
代碼如下所示:
class Base{ public: virtual ~Base()=default; private: int x; }; class A : public Base{ private: int y; }; int main () { Base *pBase = new A; delete pBase; return 0; }
C++11之前,delete
是和new
配對(duì)使用的,釋放程序在堆上開(kāi)辟得空間,將資源返還給操作系統(tǒng),C++11之后,delete又多了一個(gè)含義既:禁用成員函數(shù)的使用。使用方法為:在函數(shù)名稱后面加上=delete。
下面代碼定義了一個(gè)類(lèi),類(lèi)里面定義了一個(gè)整型的成員變量,在main函數(shù)中使用時(shí),創(chuàng)建了兩個(gè)類(lèi)的實(shí)例,一個(gè)傳入?yún)?shù)用整型一個(gè)用浮點(diǎn)型,
代碼如下:
class Test { public: Test()=delete; Test(int a):x(a) {std::cout<<x<<std::endl;} private: int x; }; int main() { Test test1(1); Test test2(1.1); return 0; }
如上,這段代碼是可以編譯通過(guò)的,因?yàn)檫@段代碼在編譯時(shí)發(fā)生了隱式轉(zhuǎn)換,將浮點(diǎn)型數(shù)據(jù)轉(zhuǎn)成了整型,代價(jià)是損失了精度。代碼運(yùn)行后輸出的結(jié)果都是:1。
如果不想在傳入非整型的數(shù)據(jù)時(shí)編譯通過(guò),就可以使用=delete
來(lái)抑制這種問(wèn)題的產(chǎn)生。如使用=delete解決上面的問(wèn)題,
代碼如下:
class Test { public: Test(int a):x(a) {std::cout<<x<<std::endl;} Test(double)=delete; private: int x; }; int main() { Test test1(1); Test test2(1.1); return 0;
編譯時(shí)報(bào)錯(cuò)信息為:
main.cpp:23:19: error: use of deleted function ‘Test::Test(double)' 23 | Test test2(1.1); | ^ main.cpp:17:5: note: declared here 17 | Test(double)=delete; | ^~~~
如上可知,使用=delete
后,可以使我們禁用一些不需要編譯器生成的默認(rèn)函數(shù),還可以避免因?yàn)閿?shù)據(jù)類(lèi)型原因?qū)е碌腻e(cuò)誤的函數(shù)調(diào)用。
到此,關(guān)于“C++中 ‘=default ’及‘ =delete ’如何使用”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識(shí),請(qǐng)繼續(xù)關(guān)注億速云網(wǎng)站,小編會(huì)繼續(xù)努力為大家?guī)?lái)更多實(shí)用的文章!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。