C++11:為何引入noexcept替代throw-創(chuàng)新互聯(lián)

引言

noexcept是C++11新引入的異常關(guān)鍵字。noexcept既是說明符,也是運(yùn)算符。作為說明符,它的作用是指定函數(shù)是否拋出異常;作為運(yùn)算符,它的作用是編譯時(shí)檢查,如果表達(dá)式不會(huì)拋出任何異常則返回 true,否則返回false。

成都創(chuàng)新互聯(lián)從2013年創(chuàng)立,是專業(yè)互聯(lián)網(wǎng)技術(shù)服務(wù)公司,擁有項(xiàng)目網(wǎng)站設(shè)計(jì)、網(wǎng)站制作網(wǎng)站策劃,項(xiàng)目實(shí)施與項(xiàng)目整合能力。我們以讓每一個(gè)夢(mèng)想脫穎而出為使命,1280元鎮(zhèn)原做網(wǎng)站,已為上家服務(wù),為鎮(zhèn)原各地企業(yè)和個(gè)人服務(wù),聯(lián)系電話:13518219792noexcept說明符

noexcept說明符的語法格式:

noexcept  // noexcept 等價(jià)于noexcept(true)
noexcept(expression) // expression可轉(zhuǎn)換為bool的常量表達(dá)式,expression為true表示函數(shù)不會(huì)拋出異常
throw() // C++11 deprecated,C++20 removed throw()等價(jià)于noexcept(true)

noexcept(false)表示允許拋出異常;noexcept(true) 表示不允許拋出異常,noexcept與noexcept(true)等價(jià);標(biāo)記了noexcept(true) 或noexcept的函數(shù)如果拋出異常了,那么std::terminate()將會(huì)調(diào)用并結(jié)束進(jìn)程。

C++11規(guī)定,滿足下述條件的函數(shù)允許拋出異常:

  • 使用throw聲明異常,除throw()外其他形式均允許拋出異常,例如,int func() throw(int)允許拋出異常,而int func() throw()不允許拋出異常。
  • noexcept(expression)中expression求值為false的函數(shù)。例如:void f() noexcept(false)允許拋出異常。
  • 函數(shù)聲明中沒有noexcept說明符的函數(shù),例如:void g() 允許拋出異常。

同時(shí)C++11也規(guī)定,滿足下述條件的函數(shù)不允許拋出異常:

  • 析構(gòu)函數(shù)(如果基類或成員允許拋出異常,析構(gòu)函數(shù)會(huì)轉(zhuǎn)為允許拋出異常)
  • 隱式聲明(或=default)的默認(rèn)構(gòu)造函數(shù),拷貝構(gòu)造函數(shù),移動(dòng)構(gòu)造函數(shù)(如果基類同名函數(shù)或成員允許拋出異常,此函數(shù)會(huì)轉(zhuǎn)換為拋出異常)
  • 隱式聲明(或=default)的拷貝賦值運(yùn)算符,移動(dòng)賦值運(yùn)算符
  • operator delete, operator delete[]

除此之外,需要特別注意的是noexcept(expression) 中的expression必須編譯期間可求值,編譯器最終會(huì)將expression 的結(jié)果轉(zhuǎn)換為 true 或false再傳遞給 noexcept(),因而可以認(rèn)為noexcept(expression) 是被轉(zhuǎn)換為 noexcept(true) 或 noexcept(false) 作用于函數(shù)聲明。

C++17之前的標(biāo)準(zhǔn)中,noexcept說明符和throw是函數(shù)簽名的一部分,但是C++17之后的標(biāo)準(zhǔn)規(guī)定中,noexcept說明符不是函數(shù)簽名的一部分。

noexcept運(yùn)算符

noexcept 運(yùn)算符會(huì)在編譯時(shí)檢查,如果表達(dá)式不會(huì)拋出任何異常則返回 true。它亦可用于函數(shù)模板的 noexcept 說明符中;noexcept運(yùn)算符的語法格式:

noexcept(expression)

noexcept(expression)返回bool型純右值。noexcept 運(yùn)算符會(huì)在編譯時(shí)檢查,編譯期不會(huì)對(duì)expression表達(dá)式進(jìn)行求值。如果expression的潛在異常集合為空,noexcept(expression)會(huì)求值為true,否則noexcept(expression)求值為false。

#include#include#includevoid mayThrow();
void noThrow() noexcept;
auto layThrow = []{};
auto lnoThrow = []() noexcept {};
 
class T
{public:
    ~T(){} 
};
 
class U
{public:
    ~U(){}
    std::vectorv;
};
 
class V
{public:
    std::vectorv;
};
 
int main()
{T t;
    U u;
    V v;
 
    std::cout<< std::boolalpha<< "mayThrow() 可能會(huì)拋出異常嗎?"<< !noexcept(mayThrow())<< '\n'<< "noThrow() 可能會(huì)拋出異常嗎?"<< !noexcept(noThrow())<< '\n'<< "lmayThrow() 可能會(huì)拋出異常嗎?"<< !noexcept(lmayThrow())<< '\n'<< "lnoThrow() 可能會(huì)拋出異常嗎?"<< !noexcept(lnoThrow())<< '\n'<< "~T() 可能會(huì)拋出異常嗎?"<< !noexcept(std::declval().~T())<< '\n'<< "T(T 右值) 可能會(huì)拋出異常嗎?"<< !noexcept(T(std::declval()))<< '\n'<< "T(T 左值) 可能會(huì)拋出異常嗎?"<< !noexcept(T(t))<< '\n'<< "U(U 右值) 可能會(huì)拋出異常嗎?"<< !noexcept(U(std::declval()))<< '\n'<< "U(U 左值) 可能會(huì)拋出異常嗎?"<< !noexcept(U(u))<< '\n' << "V(V 右值) 可能會(huì)拋出異常嗎?"<< !noexcept(V(std::declval()))<< '\n'<< "V(V 左值) 可能會(huì)拋出異常嗎?"<< !noexcept(V(v))<< '\n';  
}

輸出:

mayThrow() 可能會(huì)拋出異常嗎?true
noThrow() 可能會(huì)拋出異常嗎?false
lmayThrow() 可能會(huì)拋出異常嗎?true
lnoThrow() 可能會(huì)拋出異常嗎?false
~T() 可能會(huì)拋出異常嗎?false
T(T 右值) 可能會(huì)拋出異常嗎?false
T(T 左值) 可能會(huì)拋出異常嗎?false
U(U 右值) 可能會(huì)拋出異常嗎?true
U(U 左值) 可能會(huì)拋出異常嗎?true
V(V 右值) 可能會(huì)拋出異常嗎?false
V(V 左值) 可能會(huì)拋出異常嗎?true
noexcept的優(yōu)勢(shì)

與C++98中的throw相對(duì)比,noexcept會(huì)給我們帶來許多優(yōu)勢(shì)。

改善程序執(zhí)行效率

如果你的程序不允許拋出異常,請(qǐng)聲明為noexcept。編譯在編程鏈接生成可執(zhí)行程序時(shí),如果一個(gè)函數(shù)運(yùn)行拋出異常,編譯器會(huì)默認(rèn)值在函數(shù)末尾添加異常處理相關(guān)函數(shù)代碼段,這個(gè)代碼函數(shù)名稱一般較unwinding,所以也稱unwinding stack。stackoverflow技術(shù)網(wǎng)站的有詳細(xì)的性能評(píng)估。大家可以參考此wiki:https://stackoverflow.com/questions/26079903/noexcept-stack-unwinding-and-performance

noexcept可捕獲更多異常

noexcept可處理以前throw無法捕獲的異常,例如

struct{void CreateOtherClass() {T t{}; }
};

CreateOtherClass是否拋出異常將由T的構(gòu)造函數(shù)決定,我們可以通過這種方式實(shí)現(xiàn)。

struct{void CreateOtherClass() noexcept(is_nothrow_default_constructible::value) {T t{}; }
};

但是這個(gè)throw是無法做到的。

noexcept與throw結(jié)束異常方式不同

throw聲明函數(shù),如果拋出異常std::unexpected()將會(huì)被調(diào)用并結(jié)束進(jìn)程。noexcept聲明的函數(shù),如果拋出異常std::terminate()將會(huì)調(diào)用并結(jié)束進(jìn)程。

總結(jié)

做為C++11新引入的關(guān)鍵字,noexcept既是說明符,也是運(yùn)算符。作為說明符,它的作用是指定函數(shù)是否拋出異常;作為運(yùn)算符,它的作用是編譯時(shí)檢查,表達(dá)式不會(huì)拋出任何異常則返回 true,否則返回false。
一般情況下如果函數(shù)不會(huì)拋出異常最好添加noexcept說明符,這樣可以提升函數(shù)執(zhí)行效率,而且優(yōu)先使用noexcept替代throw。

你是否還在尋找穩(wěn)定的海外服務(wù)器提供商?創(chuàng)新互聯(lián)www.cdcxhl.cn海外機(jī)房具備T級(jí)流量清洗系統(tǒng)配攻擊溯源,準(zhǔn)確流量調(diào)度確保服務(wù)器高可用性,企業(yè)級(jí)服務(wù)器適合批量采購,新人活動(dòng)首月15元起,快前往官網(wǎng)查看詳情吧

網(wǎng)站標(biāo)題:C++11:為何引入noexcept替代throw-創(chuàng)新互聯(lián)
當(dāng)前網(wǎng)址:http://muchs.cn/article6/cddhig.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供用戶體驗(yàn)小程序開發(fā)、云服務(wù)器、App開發(fā)、營銷型網(wǎng)站建設(shè)全網(wǎng)營銷推廣

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)