C++ 中try finally關(guān)鍵字詳解
try-finally語句是Microsoft對C和C++語言的擴展,它能使32位的目標(biāo)程序在異常出現(xiàn)時,有效保證一些資源能夠被及時清除,這些資源的清除任務(wù)可以包括例如內(nèi)存的釋放,文件的關(guān)閉,文件句柄的釋放等等。try-finally語句特別適合這樣的情況下使用,例如一個例程(函數(shù))中,有幾個地方需要檢測一個錯誤,并且在錯誤出現(xiàn)時,函數(shù)可能提前返回。
#include <windows.h>
#include <stdio.h>
try-finally語句的語法與try-except很類似,稍有不同的是,__finally后面沒有一個表達式,這是因為try- finally語句的作用不是用于異常處理,所以它不需要一個表達式來判斷當(dāng)前異常錯誤的種類。另外,與try-except語句類似,try- finally也可以是多層嵌套的,并且一個函數(shù)內(nèi)可以有多個try-finally語句,不管它是嵌套的,或是平行的。當(dāng)然,try-finally多層嵌套也可以是跨函數(shù)的。這里不一一列出示例,大家可以自己測試一番。
另外,對于上面示例程序的運行結(jié)果,是不是覺得有點意料之外呢?因為 __finally塊中的put(“__finally塊中”)語句也被執(zhí)行了。是的,沒錯!這就是try-finally語句最具有魔幻能力的地方,即 “不管在何種情況下,在離開當(dāng)前的作用域時,finally塊區(qū)域內(nèi)的代碼都將會被執(zhí)行到”。呵呵!這的確是很厲害吧!為了驗證這條規(guī)則,下面來看一個更典型示例,代碼如下:
#include <stdio.h> void main() { puts(“hello”); __try { puts(“__try塊中”); // 注意,下面return語句直接讓函數(shù)返回了 return; } __finally { puts(“__finally塊中”); } puts(“world”); } 上面的程序運行結(jié)果如下: hello __try塊中 __finally塊中 Press any key to continue void main() { puts(“hello”); __try { puts(“__try塊中”); } // 注意,這里不是__except塊,而是__finally取代 __finally { puts(“__finally塊中”); } puts(“world”); }
上面的程序運行結(jié)果如下:
hello
__try塊中
__finally塊中
world
Press any key to continue
總結(jié)__finally塊被執(zhí)行的流程時,無外乎三種情況。第一種就是順序執(zhí)行到__finally塊區(qū)域內(nèi)的代碼,這種情況很簡單,容易理解;第二種就是goto語句或return語句引發(fā)的程序控制流離開當(dāng)前__try塊作用域時,系統(tǒng)自動完成對__finally塊代碼的調(diào)用;第三種就是由于在__try塊中出現(xiàn)異常時,導(dǎo)致程序控制流離開當(dāng)前__try塊作用域,這種情況下也是由系統(tǒng)自動完成對__finally塊的調(diào)用。無論是第 2種,還是第3種情況,毫無疑問,它們都會引起很大的系統(tǒng)開銷,編譯器在編譯此類程序代碼時,它會為這兩種情況準(zhǔn)備很多的額外代碼。一般第2種情況,被稱為“局部展開(LocalUnwinding)”;第3種情況,被稱為“全局展開(GlobalUnwinding)”。在后面闡述SEH實現(xiàn)的時候會詳細分析到這一點。
第3種情況,也即由于出現(xiàn)異常而導(dǎo)致的“全局展開”,對于程序員而言,這也許是無法避免的,因為你在利用異常處理機制提高程序可靠健壯性的同時,不可避免的會引起性能上其它的一些開銷。呵呵!這世界其實也算瞞公平的,有得必有失。
但是,對于第2種情況,程序員完全可以有效地避免它,避免“局部展開”引起的不必要的額外開銷。實際這也是與結(jié)構(gòu)化程序設(shè)計思想相一致的,也即一個程序模塊應(yīng)該只有一個入口和一個出口,程序模塊內(nèi)盡量避免使用goto語句等。但是,話雖如此,有時為了提高程序的可讀性,程序員在編寫代碼時,有時可能不得不采用一些與結(jié)構(gòu)化程序設(shè)計思想相悖的做法,例如,在一個函數(shù)中,可能有多處的return語句。針對這種情況,SEH提供了一種非常有效的折衷方案,那就是__leave關(guān)鍵字所起的作用,它既具有像goto語句和return語句那樣類似的作用(由于檢測到某個程序運行中的錯誤,需要馬上離開當(dāng)前的 __try塊作用域),但是又避免了“局部展開” 的額外開銷。還是看個例子吧!代碼如下:
#include <stdio.h> void test() { puts(“hello”); __try { int* p; puts(“__try塊中”); // 直接跳出當(dāng)前的__try作用域 __leave; p = 0; *p = 25; } __finally { // 這里會被執(zhí)行嗎?當(dāng)然 puts(“__finally塊中”); } puts(“world”); } void main() { __try { test(); } __except(1) { puts(“__except塊中”); } }
上面的程序運行結(jié)果如下:
hello
__try塊中
__finally塊中
world
Press any key to continue
以上所述是小編給大家介紹的C++ 中try finally關(guān)鍵字的相關(guān)知識,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
相關(guān)文章
C語言編程簡單卻重要的數(shù)據(jù)結(jié)構(gòu)順序表全面講解
這篇文章主要為大家介紹了C語言編程中非常簡單卻又非常重要的數(shù)據(jù)結(jié)構(gòu)順序表的全面講解,有需要的朋友可以借鑒參考下,希望能夠有所幫助2021-10-10一篇文章帶你了解C++Primer學(xué)習(xí)日記--處理數(shù)據(jù)
今天小編就為大家分享一篇關(guān)于C++對數(shù)器的使用講解,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2021-08-08