源码网商城,靠谱的源码在线交易网站 我的订单 购物车 帮助

源码网商城

浅析C++中boost.variant的几种访问方法

  • 时间:2021-02-21 00:14 编辑: 来源: 阅读:
  • 扫一扫,手机访问
摘要:浅析C++中boost.variant的几种访问方法
[b]Boost.Variant[/b] Variant库包含一个不同于union的泛型类,用于在存储和操作来自于不同类型的对象。这个库的一个特点是支持类型安全的访问,减少了不同数据类型的类型转换代码的共同问题。 [b]Variant 库如何改进你的程序?[/b]      •对用户指定的多种类型的进行类型安全的存储和取回      •在标准库容器中存储不同类型的方法      •变量访问的编译期检查      •高效的、基于栈的变量存储 Variant 库关注的是对一组限定类型的类型安全存储及取回,即非无类的联合。[code]Boost.Variant [/code]库与[code] Boost.Any [/code]有许多共同之外,但在功能上也有不同的考虑。在每天的编程中通常都会需要用到非无类的联合(不同的类型)。保持类型安全的一个典型方法是使用抽象基类,但这不总是可以做到的;即使可以做得,堆分配和虚拟函数的代价也可能太高。你也可以尝试用不安全的无类类型,如 void* (它会导致不幸),或者是类型安全得无限制的可变类型,如 [code]Boost.Any[/code]. 这里我们将看到 [code]Boost.Variant[/code],它支持限定的可变类型,即元素来自于一组支持的类型。 下面将浅谈variant的几种访问方法,一起来学习学习吧。 [b]使用boost::get[/b]
boost::variant<int, std::string> v;
v = "Hello world";
std::cout << boost::get<std::string>(v) << std::endl;
使用[code]boost::get[/code]来访问,需要给出原始类型,并且这样做不安全,若类型错误,程序将会抛出异常。 [b]使用RTTI[/b]
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会对程序性能有一定影响。 [b]使用访问者模式[/b]
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;
}
使用该模式,需要定义一个类并继承于[code]boost::static_visitor[/code],在类里面需要重载()操作符,通过[code]boost::apply_visitor[/code]来访问原始类型的值,这样做还是有些繁琐,每增加一个类型,都需要在var_visitor里面增加一个函数,但比使用RTTI里面的修改if-else结构好得多,因为使用访问者模式至少是遵循开放-封闭原则的,即对写开放,对修改封闭。 [b]使用模板函数[/b]
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;
}
将[code]operator()[/code]改成了模板函数的好处就是不用关心variant支持多少类型。 [b]总结[/b] 以上就是这篇文章的全部内容,希望本文的内容对大家学习或者使用C++能有所帮助,如果有疑问大家可以留言交流。谢谢大家对编程素材网的支持。
  • 全部评论(0)
联系客服
客服电话:
400-000-3129
微信版

扫一扫进微信版
返回顶部