淺析C++中boost.variant的幾種訪問方法
Boost.Variant
Variant庫包含一個不同于union的泛型類,用于在存儲和操作來自于不同類型的對象。這個庫的一個特點是支持類型安全的訪問,減少了不同數(shù)據(jù)類型的類型轉(zhuǎn)換代碼的共同問題。
Variant 庫如何改進你的程序?
•對用戶指定的多種類型的進行類型安全的存儲和取回
•在標(biāo)準(zhǔn)庫容器中存儲不同類型的方法
•變量訪問的編譯期檢查
•高效的、基于棧的變量存儲
Variant 庫關(guān)注的是對一組限定類型的類型安全存儲及取回,即非無類的聯(lián)合。Boost.Variant
庫與 Boost.Any
有許多共同之外,但在功能上也有不同的考慮。在每天的編程中通常都會需要用到非無類的聯(lián)合(不同的類型)。保持類型安全的一個典型方法是使用抽象基類,但這不總是可以做到的;即使可以做得,堆分配和虛擬函數(shù)的代價也可能太高。你也可以嘗試用不安全的無類類型,如 void* (它會導(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
來訪問,需要給出原始類型,并且這樣做不安全,若類型錯誤,程序?qū)伋霎惓!?/p>
使用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ù)可以避免類型訪問錯誤而程序異常的情況,但是這樣做有點不優(yōu)雅,每增加一個類型,都需要修改if-else結(jié)構(gòu),并且使用RTTI會對程序性能有一定影響。
使用訪問者模式
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; }
使用該模式,需要定義一個類并繼承于boost::static_visitor
,在類里面需要重載()操作符,通過boost::apply_visitor
來訪問原始類型的值,這樣做還是有些繁瑣,每增加一個類型,都需要在var_visitor里面增加一個函數(shù),但比使用RTTI里面的修改if-else結(jié)構(gòu)好得多,因為使用訪問者模式至少是遵循開放-封閉原則的,即對寫開放,對修改封閉。
使用模板函數(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)容對大家學(xué)習(xí)或者使用C++能有所幫助,如果有疑問大家可以留言交流。謝謝大家對腳本之家的支持。
相關(guān)文章
淺析C++?atomic?和?memory?ordering
這篇文章主要介紹了C++?atomic?和?memory?ordering的相關(guān)知識,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-04-04