设为首页收藏本站
网站公告 | 这是第一条公告
     

 找回密码
 立即注册
缓存时间13 现在时间13 缓存数据 风骨神仙籍里人,诗狂酒圣且平生。开元一遇成何事,留得千秋万古名。

风骨神仙籍里人,诗狂酒圣且平生。开元一遇成何事,留得千秋万古名。 -- 杨花落尽子规啼

查看: 895|回复: 2

C++中Boost的转换函数

[复制链接]

  离线 

TA的专栏

  • 打卡等级:热心大叔
  • 打卡总天数:230
  • 打卡月天数:1
  • 打卡总奖励:3531
  • 最近打卡:2025-07-02 11:35:37
等级头衔

等級:晓枫资讯-上等兵

在线时间
0 小时

积分成就
威望
0
贡献
435
主题
398
精华
0
金钱
4822
积分
896
注册时间
2023-1-8
最后登录
2025-7-2

发表于 2023-2-13 11:15:14 | 显示全部楼层 |阅读模式
Boost的转换函数是对C++中的四种类型转换函数(const_cast,reinterpret_cast,static_cast,dynamic_cast)的一些补充和扩展,在阅读本文前,请先熟悉C++中的四种类型转换函数相关知识。
polymorphic_cast

C++提供了dynamic_cast来实现运行时的类型转换,但是如果用来转换指针时,需要记得检查返回值(这是很多程序员容易忘掉的地方),否则一旦转换失败,将获得一个NULL指针,无异于给程序埋下了一个定时炸弹。
Boost的polymorphic_cast在dynamic_cast的基础上增加了对返回值的检测,如果转换失败,它就会抛出std::bad_cast异常。其函数体如下:
  1. template <class Target, class Source>
  2. inline Target polymorphic_cast(Source* x BOOST_EXPLICIT_DEFAULT_TARGET)
  3. {
  4.     Target tmp = dynamic_cast<Target>(x);
  5.     if ( tmp == 0 ) throw std::bad_cast();
  6.     return tmp;
  7. }
复制代码
虽然抛异常增加了开销,但使用起来却更加简单了。
polymorphic_downcast

由于抛出异常会降低程序的效率,而且dynamic_cast更会查询一个type_info结构来确定正确的类型,所以不管是空间上的成本还是时间上的成本,都会大大增加。在一些应用场景中,只需要在编译期间进行类型转换即可。这时我们可以使用static_cast来实现编译期间的类型转换,但static_cast可能导致错误的类型转换:
  1. struct A
  2. {
  3.     virtual ~A(){}
  4. };

  5. class B:public A{};
  6. class C:public A{};

  7. int main()
  8. {
  9.     A *pa = new C();
  10.     B *pb = static_cast<B*>(pa);
  11. }
复制代码
对于上述程序,虽然pa和pb间没有继承关系,但是这个转换却可以通过,运行时也不会报任何错误,可一旦对pb进行访问,就会得到错误的结果甚至直接导致程序死掉。
polymorphic_downcast就巧妙的解决的这一问题,首先还是先看看它的定义:
  1. template <class Target, class Source>
  2. inline Target polymorphic_downcast(Source* x BOOST_EXPLICIT_DEFAULT_TARGET)
  3. {
  4.     BOOST_ASSERT( dynamic_cast<Target>(x) == x ); // detect logic error
  5.     return static_cast<Target>(x);
  6. }
复制代码
从它的定义可以看出,在运行Release模式下,它和是static_cast一样的,也就是说它的Release版具有和static_cast一样的开销。但在Debug模式下,它会首先进行一次动态转换,而一旦类型不匹配,就会抛出异常。
在上述程序中,如果用polymorphic_downcast来替换static_cast的话,我们可以先在Debug模式下运行程序,如果有错误的类型转换,将很容易的检测出来。待改正所有的错误后,再发布Release版,这样即没有动态转换造成的开销,又杜绝了错误的类型转换。
boost::numeric_cast

在c++中,我们经常需要把不同类型的数字互相转换,如将一个数字在long和short之间转换。但由于各数字的精度不同,当一个数字从"大"类型到"小"类型就可能导致转换失败,如下所示:
  1. long n1 = 99999999;
  2. short n2 = static_cast<short>(n1);
复制代码
对于如上转换,n2得到的是一个负数,显然这个不是我们所期望的,并且这种运行时的错误是很难检测的,一旦使用了这个错误的转换后的数据,后果不堪设想。
boost::numeric_cast可以帮助我们解决这一问题,对于上面的转换,boost::numeric_cast会抛出一个boost:: bad_numeric_cast这个异常对象。从而保证转换后值的有效性。上述代码可以改写为如下:
  1. try
  2. {
  3.     long n1 = 99999999;
  4.     short n2 = boost::numeric_cast<short>(n1);
  5. }
  6. catch(boost::bad_numeric_cast&)
  7. {
  8.     std::cout<<"The conversion failed"<<std::endl;
  9. }
复制代码
numeric_cast是如何知道这样的数字转换失败的呢?numeric_cast合理的应用了std::numeric_limits<>,而std::numeric_limits<>就是内建数字类型的type_tratis。当然也可以将自己定义的数字抽象类型添加到std::numeric_limits<>的特化版本中,这样numeric_cast就可以作用于自定义的类型了。由于相对复杂点,本文是介绍其功能和用法,就不分析其源码了,感兴趣的朋友可以参看boost文档和代码。
对于numeric_cast的使用也是有些要求的。

  • 源类型和目标类型必须都是可拷贝构造的
  • 源类型和目标类型必须都是数字型类型。也就是被std::numeric_limits<>::is_specialized的特化定义为true
  • 源类型必须能被static_cast转换为目标类型
其实对我们用的系统内置的数字来说,这几条都不是限制,只有我们在需要通过它转换自定义的数据类型时,才需要注意,否则编译不通过(其实这个错误还比较好发现和解决)。
boost::lexical_cast

在C/C++程序开发中,往往需要将数字型对象的值转换为字符文本格式,或反之操作。虽然C语言就提供了不少系统函数来进行这种操作,如scanf、atoi等。这些函数小巧简洁,使用很方便,但缺少扩展性。在std中引入了stringstream来以一个通用的方式实现各种转换,但缺少对错误转换的检测。而boost::lexical_cast是在stringstream上的一个扩展,增加了对错误的类型转换的检测:
  1. #include <string>
  2. #include <iostream>
  3. #include <boost/lexical_cast.hpp>

  4. using namespace std;

  5. int main()
  6. {     
  7.     try
  8.     {
  9.         int i = 100;
  10.         string str = boost::lexical_cast<string>(i);
  11.         cout<<"The string is:"<<str<<endl;
  12.         str = "error";
  13.         i = boost::lexical_cast<int>(str);
  14.     }
  15.     catch(boost::bad_lexical_cast& exobj)
  16.     {
  17.         cout<<"Convert err:"<<endl;
  18.         cout<<exobj.what()<<endl;
  19.     }
  20. }
复制代码
在上述转换中,第二个转换从"err"到int的转换是失败的,会抛出一个boost::bad_lexical_cast的异常,从而能帮助我们构造更安全稳定的程序。
boost::lexical_cast内部实现其实也是一个stringstream的封装,其函数简化如下:
  1. template<typename Target,typename Source>
  2. Target lexical_cast(Source arg){
  3.     detail::lexical_stream<Target,Source> interpreter;
  4.     Target result;
  5.     if(!(interpreter<<arg && interpreter>>result))
  6.         throw_exception(bad_lexical_cast(typeid(Target),typeid(Source)));
  7.     return result;
  8. }
复制代码
其中lexical_stream<>对字符串流做了一系列的包装,主要提供了operator<<(Source)和operator>>(Target)操作,用于判断操作是否成功。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持晓枫资讯。

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
晓枫资讯-科技资讯社区-免责声明
免责声明:以上内容为本网站转自其它媒体,相关信息仅为传递更多信息之目的,不代表本网观点,亦不代表本网站赞同其观点或证实其内容的真实性。
      1、注册用户在本社区发表、转载的任何作品仅代表其个人观点,不代表本社区认同其观点。
      2、管理员及版主有权在不事先通知或不经作者准许的情况下删除其在本社区所发表的文章。
      3、本社区的文章部分内容可能来源于网络,仅供大家学习与参考,如有侵权,举报反馈:点击这里给我发消息进行删除处理。
      4、本社区一切资源不代表本站立场,并不代表本站赞同其观点和对其真实性负责。
      5、以上声明内容的最终解释权归《晓枫资讯-科技资讯社区》所有。
http://bbs.yzwlo.com 晓枫资讯--游戏IT新闻资讯~~~

  离线 

TA的专栏

等级头衔

等級:晓枫资讯-列兵

在线时间
0 小时

积分成就
威望
0
贡献
0
主题
0
精华
0
金钱
11
积分
2
注册时间
2024-8-25
最后登录
2024-8-25

发表于 2025-2-17 23:13:31 | 显示全部楼层
顶顶更健康!!!
http://bbs.yzwlo.com 晓枫资讯--游戏IT新闻资讯~~~

  离线 

TA的专栏

等级头衔

等級:晓枫资讯-列兵

在线时间
0 小时

积分成就
威望
0
贡献
0
主题
0
精华
0
金钱
24
积分
28
注册时间
2022-12-25
最后登录
2022-12-25

发表于 2025-4-22 05:18:35 | 显示全部楼层
感谢楼主分享。
http://bbs.yzwlo.com 晓枫资讯--游戏IT新闻资讯~~~
严禁发布广告,淫秽、色情、赌博、暴力、凶杀、恐怖、间谍及其他违反国家法律法规的内容。!晓枫资讯-社区
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

1楼
2楼
3楼

手机版|晓枫资讯--科技资讯社区 本站已运行

CopyRight © 2022-2025 晓枫资讯--科技资讯社区 ( BBS.yzwlo.com ) . All Rights Reserved .

晓枫资讯--科技资讯社区

本站内容由用户自主分享和转载自互联网,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责。

如有侵权、违反国家法律政策行为,请联系我们,我们会第一时间及时清除和处理! 举报反馈邮箱:点击这里给我发消息

Powered by Discuz! X3.5

快速回复 返回顶部 返回列表