C++ Boost Conversion超详细讲解

 更新时间:2022年11月24日 14:35:31   作者:无水先生  
Boost是为C++语言标准库提供扩展的一些C++程序库的总称。Boost库是一个可移植、提供源代码的C++库,作为标准库的后备,是C++标准化进程的开发引擎之一,是为C++语言标准库提供扩展的一些C++程序库的总称

一、说明

Boost.Conversion 在头文件 boost/cast.hpp 中定义了转换运算符 boost::polymorphic_cast 和 boost::polymorphic_downcast。它们旨在更精确地处理类型转换——通常使用 dynamic_cast 完成。

库由两个文件组成。分别在boost/cast.hpp文件中定义了boost::polymorphic_castboost::polymorphic_downcast这两个类型转换操作符, 在boost/lexical_cast.hpp文件中定义了boost::lexical_cast

二、示例和代码

boost::polymorphic_castboost::polymorphic_downcast是为了使原来用dynamic_cast实现的类型转换更加具体。具体细节,如下例所示。

struct father 
{ 
  virtual ~father() { }; 
}; 
struct mother 
{ 
  virtual ~mother() { }; 
}; 
struct child : 
  public father, 
  public mother 
{ 
}; 
void func(father *f) 
{ 
  child *c = dynamic_cast<child*>(f); 
} 
int main() 
{ 
  child *c = new child; 
  func(c); 
  father *f = new child; 
  mother *m = dynamic_cast<mother*>(f); 
} 

本例使用dynamic_cast类型转换操作符两次: 在func()函数中,它将指向父类的指针转换为指向子类的指针。在main()中, 它将一个指向父类的指针转为指向另一个父类的指针。第一个转换称为向下转换(downcast),第二个转换称为交叉转换(cross cast)。

通过使用 Boost.Conversion 的类型转换操作符,可以将向下转换和交叉转换区分开来。

#include <boost/cast.hpp> 
struct father 
{ 
  virtual ~father() { }; 
}; 
struct mother 
{ 
  virtual ~mother() { }; 
}; 
struct child : 
  public father, 
  public mother 
{ 
}; 
void func(father *f) 
{ 
  child *c = boost::polymorphic_downcast<child*>(f); 
} 
int main() 
{ 
  child *c = new child; 
  func(c); 
  father *f = new child; 
  mother *m = boost::polymorphic_cast<mother*>(f); 
} 

boost::polymorphic_downcast
 类型转换操作符只能用于向下转换。 它内部使用 
static_cast
 实现类型转换。 由于 
static_cast
 并不动态检查类型转换是否合法,所以 
boost::polymorphic_downcast
 应该只在类型转换是安全的情况下使用。 在调试(debug builds)模式下, 
boost::polymorphic_downcast
 实际上在 
assert ()
函数中使用
 dynamic_cast
 验证类型转换是否合法。 请注意这种合法性检测只在定义了
NDEBUG
宏的情况下执行,这通常是在调试模式下。

向下转换最好使用boost::polymorphic_downcast, 那么boost::polymorphic_cast就是交叉转换所需要的了。 由于dynamic_cast是唯一能实现交叉转换的类型转换操作符,boost::polymorphic_cast内部使用了它。 由于boost::polymorphic_cast能够在错误的时候抛出std::bad_cast类型的异常,所以优先使用这个类型转换操作符还是很有必要的。相反,dynamic_cast在类型转换失败使将返回0。 避免手工验证返回值,boost::polymorphic_cast提供了自动化的替代方式。

boost::polymorphic_downcastboost::polymorphic_cast只在指针必须转换的时候使用;否则,必须使用dynamic_cast执行转换。 由于boost::polymorphic_downcast是基于static_cast,所以它不能够,比如说,将父类对象转换为子类对象。 如果转换的类型不是指针,则使用boost::polymorphic_cast执行类型转换也没有什么意义,而在这种情况下使用dynamic_cast还会抛出一个std::bad_cast异常。

虽然所有的类型转换都可用dynamic_cast实现,可boost::polymorphic_downcastboost::polymorphic_cast也不是真正随意使用的。 Boost.Conversion 还提供了另外一种在实践中很有用的类型转换操作符。 体会一下下面的例子。

#include <boost/lexical_cast.hpp> 
#include <string> 
#include <iostream> 
int main() 
{ 
  std::string s = boost::lexical_cast<std::string>(169); 
  std::cout << s << std::endl; 
  double d = boost::lexical_cast<double>(s); 
  std::cout << d << std::endl; 
} 

类型转换操作符boost::lexical_cast可将数字转换为其他类型。 例子首先将整数169转换为字符串,然后将字符串转换为浮点数。

boost::lexical_cast内部使用流(streams)执行转换操作。 因此,只有那些重载了operator<<()operator>>()这两个操作符的类型可以转换。 使用boost::lexical_cast的优点是类型转换出现在一行代码之内,无需手工操作流(streams)。 由于流的用法对于类型转换不能立刻理解代码含义, 而boost::lexical_cast类型转换操作符还可以使代码更有意义,更加容易理解。

请注意boost::lexical_cast并不总是访问流(streams);它自己也优化了一些数据类型的转换。

如果转换失败,则抛出boost::bad_lexical_cast类型的异常,它继承自std::bad_cast

#include <boost/lexical_cast.hpp> 
#include <string> 
#include <iostream> 
int main() 
{ 
  try 
  { 
    int i = boost::lexical_cast<int>("abc"); 
    std::cout << i << std::endl; 
  } 
  catch (boost::bad_lexical_cast &e) 
  { 
    std::cerr << e.what() << std::endl; 
  } 
} 

三、更多示例代码

本例由于字符串 "abc" 不能转换为int类型的数字而抛出异常。

示例 54.1。使用 dynamic_cast 向下和交叉投射

struct base1 { virtual ~base1() = default; };
struct base2 { virtual ~base2() = default; };
struct derived : public base1, public base2 {};
void downcast(base1 *b1)
{
  derived *d = dynamic_cast<derived*>(b1);
}
void crosscast(base1 *b1)
{
  base2 *b2 = dynamic_cast<base2*>(b1);
}
int main()
{
  derived *d = new derived;
  downcast(d);
  base1 *b1 = new derived;
  crosscast(b1);
}

Example54.1

示例 54.1 两次使用了转换运算符 dynamic_cast:在 downcast() 中,它将指向基类的指针转换为指向派生类的指针。在 crosscast() 中,它将指向基类的指针转换为指向不同基类的指针。第一个转换是向下转换,第二个转换是交叉转换。 Boost.Conversion 中的转换运算符让您可以区分向下转换和交叉转换。

示例 54.2。使用 polymorphic_downcast 和 polymorphic_cast 进行向下和交叉转换

#include <boost/cast.hpp>
struct base1 { virtual ~base1() = default; };
struct base2 { virtual ~base2() = default; };
struct derived : public base1, public base2 {};
void downcast(base1 *b1)
{
  derived *d = boost::polymorphic_downcast<derived*>(b1);
}
void crosscast(base1 *b1)
{
  base2 *b2 = boost::polymorphic_cast<base2*>(b1);
}
int main()
{
  derived *d = new derived;
  downcast(d);
  base1 *b1 = new derived;
  crosscast(b1);
}

boost::polymorphic_downcast(参见示例 54.2)只能用于向下转型,因为它使用 static_cast 来执行转换。因为 static_cast 不会动态检查转换的有效性,boost::polymorphic_downcast 必须仅在转换安全时使用。在调试版本中,boost::polymorphic_downcast 使用 dynamic_cast 和 assert() 来确保类型转换有效。只有在未定义宏 NDEBUG 时才会执行此测试,这通常是调试版本的情况。

boost::polymorphic_cast 是交叉转换所必需的。 boost::polymorphic_cast 使用 dynamic_cast,它是唯一可以执行交叉转换的转换运算符。最好使用 boost::polymorphic_cast 而不是 dynamic_cast,因为前者在出现错误时抛出 std::bad_cast 类型的异常,而 dynamic_cast 在类型转换失败时返回空指针。

仅使用 boost::polymorphic_downcast 和 boost::polymorphic_cast 来转换指针;否则,使用 dynamic_cast。因为 boost::polymorphic_downcast 是基于 static_cast 的,所以它不能将基类的对象转换为派生类的对象。此外,使用 boost::polymorphic_cast 转换指针以外的类型没有意义,因为如果转换失败,dynamic_cast 将抛出 std::bad_cast 类型的异常。

到此这篇关于C++ Boost Conversion超详细讲解的文章就介绍到这了,更多相关C++ Boost Conversion内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Linux中使用VS Code编译调试C++项目详解

    Linux中使用VS Code编译调试C++项目详解

    最近因为项目的需求,需要在Linux下开发C++相关项目,经过一番摸索最终实现了,下面这篇文章就给大家简单总结了一下如何通过VS Code进行编译调试的一些注意事项。有需要的朋友们可以参考借鉴,下面来跟着小编一起看看吧。
    2016-12-12
  • C语言数据结构不挂科指南之队列详解

    C语言数据结构不挂科指南之队列详解

    这篇博客主要介绍一下队列的概念,并且采用 C 语言,编写两种存储实现方式:顺序存储和链式存储,当然还有常规的队列基本操作的实现算法
    2022-09-09
  • C语言实现SM4加解密方式

    C语言实现SM4加解密方式

    这篇文章主要介绍了C语言实现SM4加解密方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-04-04
  • C++实现String类实例代码

    C++实现String类实例代码

    这篇文章主要介绍了C++实现String类实例代码的相关资料,需要的朋友可以参考下
    2017-04-04
  • C语言实现简单学生成绩管理系统项目

    C语言实现简单学生成绩管理系统项目

    这篇文章主要为大家详细介绍了C语言实现简单学生成绩管理系统项目,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-07-07
  • c++中for双循环的那些事

    c++中for双循环的那些事

    本人很菜,今天看《C++编程思想》中的一道课后题中说到这样一个问题。修改两层嵌套的for循环的标识符,观察结果变化
    2013-05-05
  • 快速解决boost库链接出错的问题(分享)

    快速解决boost库链接出错的问题(分享)

    下面小编就为大家带来一篇快速解决boost库链接出错的问题(分享)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-05-05
  • C++ LeetCode1827题解最少操作使数组递增

    C++ LeetCode1827题解最少操作使数组递增

    这篇文章主要为大家介绍了C++ LeetCode1827题解最少操作使数组递增示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-12-12
  • C++中 set的用法

    C++中 set的用法

    这篇文章主要介绍了C++中 set的用法,set的内部使用了红黑树对所有的元素进行了排序。在树结构当中,我们通常使用的都是<key, value>的形式。下面我们来看看该内容的具体情况,需要的朋友也可以参考一下
    2021-11-11
  • 双缓冲解决VC++绘图时屏幕闪烁

    双缓冲解决VC++绘图时屏幕闪烁

    相信很多人在做图形界面开发时,常常会遇到屏幕闪烁的情况,当然我也不例外,下面我们就来详细探讨下这个问题的解决办法
    2015-08-08

最新评论