关于C++的强制类型转换浅析

发布时间 - 2026-01-11 03:11:09    点击率:

前言

一说起强制类型转换大家都很熟悉,相信很多学习完C++的朋友还在使用C语言的强制类型的方式 (类型)变量.

C++其实也具有自己的一套强制类型转换它们分明是:static_cast  reinterpret_cast  const_cast  dynamic_cast四种类型.

那么肯定会有人好奇C++是不是闲,C语言的强制类型用的舒舒服服的,为什么要新推出来这几个?

新类型的强制转换可以提供更好的控制强制转换过程,允许控制各种不同种类的强制转换。C++中风格是static_cast<type>。C++风格的强制转换其他的好处是,它们能更清晰的表明它们要干什么。程序员只要扫一眼这样的代码,就能立即知道一个强制转换的目

的。

static_cast                                                                                              

static_cast用于非多态类型的转换(静态转换),任何标准转换都可以用它,但它不能用于两个不相关的类型进行转换.

何为不相关类型? 比如int 和 double char short就是相关类型. 和int*就是不相关类型.

我们来看一看static_cast的用法.  例如,通过将一个运算对象强制转换成double类型就能使表达式执行浮点数除法:

  double slope = static_cast<double>(j) / i;

当static_cast需要把一个较大的算术类型赋值给较小的类型时,static_cast非常有用。此时,强制类型转换告诉程序的读者和编译器:我们知道并且不在乎潜在的精度损失。一般来说,如果编译器发现一个的算术类型试图赋值给较小的类型,就会给出警告信息;但是当我们执行了显式的类型转换后,警告信息就会被关闭了。

reinterpret_cast                                                                                        

reinterpret_cast有着和C风格的强制转换同样的能力。它可以转化任何内置的数据类型为其他任何的数据类型,也可以转化任何指针类型为其他的类型。它甚至可以转化内置的数据类型为指针,无须考虑类型安全或者常量的情形。不到万不得已绝对用。

因为reinterpret_cast是一个蛮bug的操作,下面我来演示一下.

typedef void (* FUNC)(); 
int DoSomething (int i) 
{ 
  cout<<"DoSomething" <<endl; 
  return 0; 
} 
void Test () 
{ 
  // reinterpret_cast可以编译器以FUNC的定义方式去看待 DoSomething函数 
  // 所以非常的BUG,下面转换函数指针的代码是不可移植的,所以不建议这样用 
  // C++不保证所有的函数指针都被一样的使用,所以这样用有时会产生不确定的结果 
  FUNC f = reinterpret_cast< FUNC>(DoSomething ); 
  f(); 
} 

当你这样运行的时候,你会发现通过函数指针没有传参数调用这个有参数的函数居然可以调用,这就很尴尬了,所以我告诉你不到万不得已就不要使用reinterpret_cast

const_cast

对于将常量对象转化为非常量对象的行为,我们一般称之为“去掉const性质”。一旦我们去掉了某个对象的const性质,编译器就不再阻止我们对该对象进行写操作了。如果对象本身不是一个常量,使用强制类型转换获得写权限是合法的行为。然而如果对象是一个常量,再使用const_cast执行写操作就会产生未定义的后果。

举个例子:

#include<iostream> 
#include<Windows.h> 
#include<assert.h> 
 
using namespace std; 
 
int main() 
{ 
 const int a = 2; 
 int *p = const_cast<int*>(&a); 
 *p = 3; 
 cout << a << endl; 
 system("pause"); 
 return 0; 
} 

我们有理由的认为在内存当中a的值被修改为3,但是结果呢? 我们来看一看

这不科学啊?? 我们再打开监视窗口看一下a的值.

我们都知道监视窗口看到的都是从内存当中拿到的,但是为什么内存当中为3,打印出来就是2呢? 我来解释一下.

C++编译器具有优化功能,当你定一个const的常量的时候,系统觉得它不会被改变了,于是做一个优化把该常量存到寄存器当中,下次访问的过程更快速一点. 所以当显示窗口读取数据的时候,他会直接去寄存器当中读取数据.而不是去内存,所以导致我们明明该掉了a的值,却打印不出来,但是如何解决这个问题呢??

c++有一个关键字: volatile 该关键字的作用防止编译器优化,这个时候要输出a就会老老实实的回内存去查看.

#include<iostream> 
#include<Windows.h> 
#include<assert.h> 
 
using namespace std; 
 
int main() 
{ 
 volatile const int a = 2; 
 int *p = const_cast<int*>(&a); 
 *p = 3; 
 cout << a << endl; 
 system("pause"); 
 return 0; 
} 

dynamic_cast                                                                                           

前三种的强制类型转换,他们能做到的C语言的强制类型转换也大多能做到,最后一种C语言的强制类型转换就没有办法了.

在类的转换时,在类层次间进行上行转换时,dynamic_cast和static_cast的效果是一样的。在进行下行转换时,dynamic_cast具有类型检查的功能,比static_cast更安全。当用于多态类型时,它允许任意的隐式类型转换以及相反过程. 不过,与static_cast不同,在后一种情况里(注:即隐式转换的相反过程),dynamic_cast 会检查操作是否有效. 也就是说, 它会检查转换是否会返回一个被请求的有效的完整对象。检测在运行时进行. 如果被转换的指针不是一个被请求的有效完整的对象指针,返回值为NULL. 对于引用 类型,会抛出bad_cast异常

你说这个强转有啥用,其实对于我这种菜鸟还真的没用过,不过我知道一个问题可以使用这样的方法解决. 给你两个类让你分辨那个是子类那个是父类,我们来看看是如何解决的.

#include<iostream> 
#include<Windows.h> 
#include<assert.h> 
 
using namespace std; 
 
class AA 
{ 
public: 
 
 virtual void fun1() 
 { 
  cout << "hehe" << endl; 
 } 
public: 
 int a; 
}; 
 
class BB :public AA 
{ 
public: 
 
 virtual void fun1() 
 { 
  cout << "heh2e" << endl; 
 } 
public: 
 int c; 
}; 
int main() 
{ 
 AA* q = new AA(); 
 BB* p = new BB(); 
 AA* a; 
 BB* b; 
 
 b = dynamic_cast<BB*>(q); 
 if (b == NULL) 
 { 
  cout << "AA为基类" << endl; 
 } 
 else{ 
  cout << "AA为子类" << endl; 
 } 
 
 a = dynamic_cast<AA*>(p); 
 if (a == NULL) 
 { 
  cout << "BB为基类" << endl; 
 } 
 else 
 { 
  cout << "BB为子类" << endl; 
 } 
 system("pause"); 
 return 0; 
} 

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对的支持。


# c  # 强制类型转换  # c语言强制类型转换  # C++强制类型转换的四种方式  # 解析C++中四种强制类型转换的区别详解  # 基于c++强制类型转换的(总结)详解  # C++中4种强制类型转换的区别总结  # C++超详细讲解强制类型转换  # 深入C++四种强制类型转换的总结  # C++ 强制类型转换详解  # 浅谈C++的语句语法与强制数据类型转换  # C++强制类型转换详细示例代码  # 就会  # 子类  # 是一个  # 我来  # 当你  # 万不得已  # 不相关  # 掉了  # 较小  # 自己的  # 让你  # 多态  # 给你  # 菜鸟  # 还在  # 隐式  # 大家都  # 就能  # 你说  # 告诉你 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 制作旅游网站html,怎样注册旅游网站?  php后缀怎么变mp4格式错误_修改扩展名提示格式不对怎么办【技巧】  laravel怎么实现图片的压缩和裁剪_laravel图片压缩与裁剪方法  Laravel怎么连接多个数据库_Laravel多数据库连接配置  网站制作价目表怎么做,珍爱网婚介费用多少?  JS经典正则表达式笔试题汇总  Laravel控制器是什么_Laravel MVC架构中Controller的作用与实践  如何用y主机助手快速搭建网站?  高端云建站费用究竟需要多少预算?  Android Socket接口实现即时通讯实例代码  如何构建满足综合性能需求的优质建站方案?  如何解决hover在ie6中的兼容性问题  javascript中的数组方法有哪些_如何利用数组方法简化数据处理  HTML5空格和margin有啥区别_空格与外边距的使用场景【说明】  jQuery 常见小例汇总  如何快速查询网址的建站时间与历史轨迹?  油猴 教程,油猴搜脚本为什么会网页无法显示?  Laravel怎么实现模型属性的自动加密  如何选择PHP开源工具快速搭建网站?  Laravel怎么处理异常_Laravel自定义异常处理与错误页面教程  Laravel如何实现API版本控制_Laravel API版本化路由设计策略  如何续费美橙建站之星域名及服务?  如何快速搭建高效香港服务器网站?  网易LOFTER官网链接 老福特网页版登录地址  Java垃圾回收器的方法和原理总结  网站制作壁纸教程视频,电脑壁纸网站?  如何做网站制作流程,*游戏网站怎么搭建?  香港网站服务器数量如何影响SEO优化效果?  laravel怎么通过契约(Contracts)编程_laravel契约(Contracts)编程方法  EditPlus 正则表达式 实战(3)  Laravel Livewire是什么_使用Laravel Livewire构建动态前端界面  制作电商网页,电商供应链怎么做?  高性能网站服务器部署指南:稳定运行与安全配置优化方案  Laravel如何从数据库删除数据_Laravel destroy和delete方法区别  Laravel如何集成微信支付SDK_Laravel使用yansongda-pay实现扫码支付【实战】  如何快速搭建安全的FTP站点?  米侠浏览器网页背景异常怎么办 米侠显示修复  百度浏览器ai对话怎么关 百度浏览器ai聊天窗口隐藏  如何使用 jQuery 正确渲染 Instagram 风格的标签列表  lovemo网页版地址 lovemo官网手机登录  实现点击下箭头变上箭头来回切换的两种方法【推荐】  谷歌浏览器如何更改浏览器主题 Google Chrome主题设置教程  免费的流程图制作网站有哪些,2025年教师初级职称申报网上流程?  javascript日期怎么处理_如何格式化输出  如何在IIS管理器中快速创建并配置网站?  大型企业网站制作流程,做网站需要注册公司吗?  如何快速搭建高效WAP手机网站?  Win11怎么开启自动HDR画质_Windows11显示设置HDR选项  如何用好域名打造高点击率的自主建站?  Laravel路由Route怎么设置_Laravel基础路由定义与参数传递规则【详解】