C++?Boost?ScopeExit超詳細(xì)講解
一、提要
資源有很多種,每種都封裝一套,還是挺繁瑣的!對(duì)于比較少使用或者一個(gè)程序很可能只會(huì)用一次的資源,我們不想封裝,在這種情況下用Boost.ScopeExit。
二、退出作用域(Boost.ScopeExit)
庫 Boost.ScopeExit 使得在沒有資源特定類的情況下使用 RAII 成為可能。
2.1 范例1.UsingBOOST_SCOPE_EXIT
#include <boost/scope_exit.hpp> #include <iostream> int *foo() { int *i = new int{10}; BOOST_SCOPE_EXIT(&i) { delete i; i = 0; } BOOST_SCOPE_EXIT_END std::cout << *i << '\n'; return i; } int main() { int *j = foo(); std::cout << j << '\n'; }
計(jì)算結(jié)果:
Boost.ScopeExit 提供了宏 BOOST_SCOPE_EXIT,它可以用來定義一些看起來像本地函數(shù)但沒有名字的東西。但是,它確實(shí)有一個(gè)括號(hào)中的參數(shù)列表和大括號(hào)中的塊。 必須包含頭文件 boost/scoped_exit.hpp 才能使用 BOOST_SCOPE_EXIT。
宏的參數(shù)列表包含來自外部范圍的變量,這些變量應(yīng)該可以在塊中訪問。變量通過副本傳遞。要通過引用傳遞變量,它必須以 & 符號(hào)作為前綴,如示例2.1 范例1. 中所示。
如果變量在參數(shù)列表中,則塊中的代碼只能從外部范圍訪問變量。
BOOST_SCOPE_EXIT 用于定義一個(gè)塊,該塊將在定義塊的范圍結(jié)束時(shí)執(zhí)行。在示例 3.1 中,使用 BOOST_SCOPE_EXIT 定義的塊在 foo() 返回之前執(zhí)行。
BOOST_SCOPE_EXIT 可用于從 RAII 中受益,而無需使用特定于資源的類。 foo() 使用 new 創(chuàng)建一個(gè) int 變量。為了釋放變量,使用 BOOST_SCOPE_EXIT 定義了一個(gè)調(diào)用 delete 的塊。即使函數(shù)由于異常而提前返回,也保證執(zhí)行此塊。在示例 1 中,BOOST_SCOPE_EXIT 與智能指針一樣好。
請(qǐng)注意,變量 i 在 BOOST_SCOPE_EXIT 定義的塊末尾設(shè)置為 0。然后 i 由 foo() 返回并寫入 main() 中的標(biāo)準(zhǔn)輸出流。但是,該示例不顯示 0。j 設(shè)置為隨機(jī)值 - 即 int 變量在內(nèi)存被釋放之前所在的地址。 BOOST_SCOPE_EXIT 后面的塊獲得了對(duì) i 的引用并釋放了內(nèi)存。但由于該塊是在 foo() 的末尾執(zhí)行的,因此將 0 分配給 i 為時(shí)已晚。 foo() 的返回值是在 i 設(shè)置為 0 之前創(chuàng)建的 i 的副本。
如果您使用 C++11 開發(fā)環(huán)境,則可以忽略 Boost.ScopeExit。在這種情況下,您可以在 lambda 函數(shù)的幫助下使用沒有資源特定類的 RAII。
2.2 示例2.Boost.ScopeExit和C++11的lambda函數(shù)
#include <iostream> #include <utility> template <typename T> struct scope_exit { scope_exit(T &&t) : t_{std::move(t)} {} ~scope_exit() { t_(); } T t_; }; template <typename T> scope_exit<T> make_scope_exit(T &&t) { return scope_exit<T>{ std::move(t)}; } int *foo() { int *i = new int{10}; auto cleanup = make_scope_exit([&i]() mutable { delete i; i = 0; }); std::cout << *i << '\n'; return i; } int main() { int *j = foo(); std::cout << j << '\n'; }
運(yùn)算結(jié)果:
示例2 定義了類 scope_exit,其構(gòu)造函數(shù)接受一個(gè)函數(shù)。該函數(shù)由析構(gòu)函數(shù)調(diào)用。此外,還定義了一個(gè)輔助函數(shù) make_scope_exit(),它可以在無需指定模板參數(shù)的情況下實(shí)例化 scope_exit。
在 foo() 中,一個(gè) lambda 函數(shù)被傳遞給 make_scope_exit()。 lambda 函數(shù)看起來像示例 3.1 中 BOOST_SCOPE_EXIT 之后的塊:地址存儲(chǔ)在 i 中的動(dòng)態(tài)分配的 int 變量通過刪除被釋放。然后將 0 分配給 i。
該示例與前一個(gè)示例執(zhí)行相同的操作。不僅 int 變量被刪除,而且 j 在寫入標(biāo)準(zhǔn)輸出流時(shí)也沒有設(shè)置為 0。
2.3 示例3.特點(diǎn)BOOST_SCOPE_EXIT
#include <boost/scope_exit.hpp> #include <iostream> struct x { int i; void foo() { i = 10; BOOST_SCOPE_EXIT(void) { std::cout << "last\n"; } BOOST_SCOPE_EXIT_END BOOST_SCOPE_EXIT(this_) { this_->i = 20; std::cout << "first\n"; } BOOST_SCOPE_EXIT_END } }; int main() { x obj; obj.foo(); std::cout << obj.i << '\n'; }
示例介紹了 BOOST_SCOPE_EXIT 的一些特性:
- 當(dāng) BOOST_SCOPE_EXIT 用于在一個(gè)范圍內(nèi)定義多個(gè)塊時(shí),這些塊以相反的順序執(zhí)行。示例 3 先顯示后顯示。
- 如果沒有變量將傳遞給 BOOST_SCOPE_EXIT,則需要指定 void。括號(hào)不能為空。
- 如果您在成員函數(shù)中使用 BOOST_SCOPE_EXIT 并且需要傳遞指向當(dāng)前對(duì)象的指針,則必須使用 this_ 而不是 this。
示例3:顯示first,last
, 和20
in that order.
三、練習(xí)
BOOST_SCOPE_EXIT(&flag) { } BOOST_SCOPE_EXIT_END
BOOST_SCOPE_EXIT和 BOOST_SCOPE_EXIT_END都是配合著使用的, BOOST_SCOPE_EXIT()里面可以傳入多個(gè)參數(shù)。其作用相當(dāng)于回調(diào)函數(shù),
在作用域結(jié)束之后程序會(huì)自動(dòng)調(diào)用 BOOST_SCOPE_EXIT到 BOOST_SCOPE_EXIT_END之間的代碼。
包含的頭文件是 #include <boost/scope_exit.hpp>
下面是一個(gè)簡單的例子:
#include <boost/scope_exit.hpp> void PrintCout(const std::string& str) { bool submitted = false; BOOST_SCOPE_EXIT(&submitted ) { if (!submitted) std::cout<<"false"<<std::endl; else std::cout<<"true"<<std::endl; } BOOST_SCOPE_EXIT_END submitted = ((str == "Y")? true:false); } int main(int argc, char* argv[]) { std::string str; std::cin>>str; while(str != "q"){ PrintCout(str); std::cin>>str; } }
這里的 BOOST_SCOPE_EXIT()可以傳入多個(gè)參數(shù),格式:
BOOST_SCOPE_EXIT(¶meter1,¶meter2,...)
到此這篇關(guān)于C++ Boost ScopeExit超詳細(xì)講解的文章就介紹到這了,更多相關(guān)C++ Boost ScopeExit內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++設(shè)計(jì)模式編程中簡單工廠與工廠方法模式的實(shí)例對(duì)比
這篇文章主要介紹了C++設(shè)計(jì)模式編程中簡單工廠與工廠方法模式的實(shí)例對(duì)比,文中最后對(duì)兩種模式的優(yōu)缺點(diǎn)總結(jié)也比較詳細(xì),需要的朋友可以參考下2016-03-03MFC創(chuàng)建模態(tài)對(duì)話框和非模態(tài)對(duì)話框的方法
這篇文章主要介紹了MFC創(chuàng)建模態(tài)對(duì)話框和非模態(tài)對(duì)話框的方法,需要的朋友可以參考下2014-07-07C++實(shí)現(xiàn)學(xué)生成績管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)學(xué)生成績管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-12-12C++ 中回調(diào)函數(shù)詳解及簡單實(shí)例
這篇文章主要介紹了C++ 中回調(diào)函數(shù)詳解及簡單實(shí)例的相關(guān)資料,需要的朋友可以參考下2017-06-06