Boost.Variant
Variant庫包含一個不同于union的泛型類,用于在存儲和操作來自于不同類型的對象。這個庫的一個特點是支持類型安全的訪問,減少了不同數據類型的類型轉換代碼的共同問題。
Variant 庫如何改進你的程序?
•對用戶指定的多種類型的進行類型安全的存儲和取回
•在標準庫容器中存儲不同類型的方法
•變量訪問的編譯期檢查
•高效的、基于棧的變量存儲
Variant 庫關注的是對一組限定類型的類型安全存儲及取回,即非無類的聯合。Boost.Variant
庫與 Boost.Any
有許多共同之外,但在功能上也有不同的考慮。在每天的編程中通常都會需要用到非無類的聯合(不同的類型)。保持類型安全的一個典型方法是使用抽象基類,但這不總是可以做到的;即使可以做得,堆分配和虛擬函數的代價也可能太高。你也可以嘗試用不安全的無類類型,如 void* (它會導致不幸),或者是類型安全得無限制的可變類型,如 Boost.Any
. 這里我們將看到 Boost.Variant
,它支持限定的可變類型,即元素來自于一組支持的類型。
下面將淺談variant的幾種訪問方法,一起來學習學習吧。
使用boost::get
1
2
3
|
boost::variant< int , std::string> v; v = "Hello world" ; std::cout << boost::get<std::string>(v) << std::endl; |
使用boost::get
來訪問,需要給出原始類型,并且這樣做不安全,若類型錯誤,程序將會拋出異常。
使用RTTI
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
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技術可以避免類型訪問錯誤而程序異常的情況,但是這樣做有點不優雅,每增加一個類型,都需要修改if-else結構,并且使用RTTI會對程序性能有一定影響。
使用訪問者模式
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
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里面增加一個函數,但比使用RTTI里面的修改if-else結構好得多,因為使用訪問者模式至少是遵循開放-封閉原則的,即對寫開放,對修改封閉。
使用模板函數
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
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()
改成了模板函數的好處就是不用關心variant支持多少類型。
總結
以上就是這篇文章的全部內容,希望本文的內容對大家學習或者使用C++能有所幫助,如果有疑問大家可以留言交流。謝謝大家對服務器之家的支持。