淺析C++中boost.variant的幾種訪問方法
Boost.Variant
Variant庫包含一個(gè)不同于union的泛型類,用于在存儲(chǔ)和操作來自于不同類型的對(duì)象。這個(gè)庫的一個(gè)特點(diǎn)是支持類型安全的訪問,減少了不同數(shù)據(jù)類型的類型轉(zhuǎn)換代碼的共同問題。
Variant 庫如何改進(jìn)你的程序?
•對(duì)用戶指定的多種類型的進(jìn)行類型安全的存儲(chǔ)和取回
•在標(biāo)準(zhǔn)庫容器中存儲(chǔ)不同類型的方法
•變量訪問的編譯期檢查
•高效的、基于棧的變量存儲(chǔ)
Variant 庫關(guān)注的是對(duì)一組限定類型的類型安全存儲(chǔ)及取回,即非無類的聯(lián)合。Boost.Variant
庫與 Boost.Any
有許多共同之外,但在功能上也有不同的考慮。在每天的編程中通常都會(huì)需要用到非無類的聯(lián)合(不同的類型)。保持類型安全的一個(gè)典型方法是使用抽象基類,但這不總是可以做到的;即使可以做得,堆分配和虛擬函數(shù)的代價(jià)也可能太高。你也可以嘗試用不安全的無類類型,如 void* (它會(huì)導(dǎo)致不幸),或者是類型安全得無限制的可變類型,如 Boost.Any
. 這里我們將看到 Boost.Variant
,它支持限定的可變類型,即元素來自于一組支持的類型。
下面將淺談variant的幾種訪問方法,一起來學(xué)習(xí)學(xué)習(xí)吧。
使用boost::get
boost::variant<int, std::string> v; v = "Hello world"; std::cout << boost::get<std::string>(v) << std::endl;
使用boost::get
來訪問,需要給出原始類型,并且這樣做不安全,若類型錯(cuò)誤,程序?qū)?huì)拋出異常。
使用RTTI
void var_print(boost::variant<int, std::string>& v) { if (v.type() == typeid(int)) { std::cout << get<int>(v) << std::endl; } else if (v.type() == typeid(std::string)) { std::cout << get<std::string>(v) << std::endl; } // Else do nothing } int main() { boost::variant<int, std::string> v; v = "Hello world"; var_print(v); return 0; }
使用RTTI技術(shù)可以避免類型訪問錯(cuò)誤而程序異常的情況,但是這樣做有點(diǎn)不優(yōu)雅,每增加一個(gè)類型,都需要修改if-else結(jié)構(gòu),并且使用RTTI會(huì)對(duì)程序性能有一定影響。
使用訪問者模式
class var_visitor : public boost::static_visitor<void> { public: void operator()(int& i) const { std::cout << i << std::endl; } void operator()(std::string& str) const { std::cout << str << std::endl; } }; int main() { boost::variant<int, std::string> v; v = "Hello world"; boost::apply_visitor(var_visitor(), v); return 0; }
使用該模式,需要定義一個(gè)類并繼承于boost::static_visitor
,在類里面需要重載()操作符,通過boost::apply_visitor
來訪問原始類型的值,這樣做還是有些繁瑣,每增加一個(gè)類型,都需要在var_visitor里面增加一個(gè)函數(shù),但比使用RTTI里面的修改if-else結(jié)構(gòu)好得多,因?yàn)槭褂迷L問者模式至少是遵循開放-封閉原則的,即對(duì)寫開放,對(duì)修改封閉。
使用模板函數(shù)
class var_visitor : public boost::static_visitor<void> { public: template<typename T> void operator()(T& i) const { std::cout << i << std::endl; } }; int main() { boost::variant<int, std::string> v; v = "Hello world"; boost::apply_visitor(var_visitor(), v); return 0; }
將operator()
改成了模板函數(shù)的好處就是不用關(guān)心variant支持多少類型。
總結(jié)
以上就是這篇文章的全部內(nèi)容,希望本文的內(nèi)容對(duì)大家學(xué)習(xí)或者使用C++能有所幫助,如果有疑問大家可以留言交流。謝謝大家對(duì)腳本之家的支持。
- C++中Boost庫裁剪與其應(yīng)用詳解
- 使用設(shè)計(jì)模式中的單例模式來實(shí)現(xiàn)C++的boost庫
- C++之BOOST字符串查找示例
- C++之Boost::array用法簡介
- C++之boost::array的用法
- c++中new的三種用法詳細(xì)解析
- c++ vector(向量)使用方法詳解(順序訪問vector的多種方式)
- 基于C++全局變量的聲明與定義的詳解
- C++運(yùn)算符重載的方法詳細(xì)解析
- C++中的friend友元函數(shù)詳細(xì)解析
- C++中的哈希容器unordered_map使用示例
- C++ boost 時(shí)間與日期處理詳細(xì)介紹
相關(guān)文章
淺析C++?atomic?和?memory?ordering
這篇文章主要介紹了C++?atomic?和?memory?ordering的相關(guān)知識(shí),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-04-04C語言實(shí)現(xiàn)520表白代碼 祝你表白成功!
這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)520表白代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-05-05linux下實(shí)現(xiàn)的2048游戲示例分享
這篇文章主要介紹了linux下實(shí)現(xiàn)的2048游戲示例,需要的朋友可以參考下2014-04-04