当前位置: 首页 > news >正文

CppCon 2015 学习:The dangers of C-style casts

C风格类型转换的危险

Joshua Gerrard 的文章讨论了 C风格类型转换(C-style cast)的危险性,并通过几个示例来演示它的潜在问题。C风格转换有时候会让程序看起来没有错误,但实际上可能会引入潜在的类型不匹配问题,这些问题可能在编译时没有警告,也可能在运行时导致错误。以下是对文章的中文理解:

示例 1:
float x = 1.0f;
double* y = (double*) &x;

你认为这段代码能通过编译吗?

  • GCC 5.2.0 和 MSVC 19 甚至不会发出警告。C风格转换 (double*) 会让编译器接受将 float* 强制转换为 double*,但是这种转换是非法的,因为 floatdouble 类型在内存布局上是不同的。
    • 问题: 即使没有警告,它仍然是不安全的,因为这样做会破坏类型系统,可能导致不可预期的行为。
      改成 static_cast 之后:
float x = 1.0f;
double* y = static_cast<double*>(&x);

你认为这会编译通过吗?

  • GCC 5.2.0 会报错:invalid static_cast from type 'float*' to type 'double*'
  • MSVC 19 会报错:C2440 'static_cast': cannot convert from 'float *' to 'double *'
    分析:
  • static_cast 是类型转换的一个更安全的版本。它在编译时会进行类型检查,如果转换无效,则会发出错误。而 C风格转换则没有这种检查,可能导致程序在运行时出现潜在的类型问题。
示例 2:
struct T;
const float x = 1.0f;
T* y = (T*) &x;

你认为这段代码能通过编译吗?

  • GCC 5.2.0 和 MSVC 19 不会发出任何警告,直接编译通过。但它实际上是错误的。
    改成 static_cast 之后:
T* y = static_cast<T*>(&x);

你认为这会编译通过吗?

  • GCC 5.2.0 会报错:invalid static_cast from type 'const float*' to type 'T*'
  • MSVC 19 会报错:C2440: 'static_cast': cannot convert from 'const float *' to 'T *'
    分析:
  • static_cast 在这里无法进行类型转换,因为 x 是一个 const 类型的对象,而 T* 是一个非 const 类型的指针。static_cast 不能在这种情况下去除 const 限定符。
  • C风格转换 不会做这种检查,它会将 const 移除,这会导致潜在的未定义行为。
发生了什么?

在 C++ 中,C风格转换是 “隐式的多重类型转换”。根据 N4296 - 5.4 Explicit type conversion 规范,C风格转换会尝试以下几个步骤,直到有一个成功:

  1. const_cast:移除或添加 const 限定符。
  2. static_cast:执行类型之间的转换(如从基类到派生类)。
  3. reinterpret_cast:执行不同类型之间的转换(通常用于低级内存操作)。
  4. reinterpret_cast + const_cast:同时进行 reinterpret_castconst_cast
    问题:
    C风格转换会进行多个步骤的类型转换,而且不会检查是否符合类型系统的规则,这就导致了它的危险性。对于复杂的类型转换,C风格转换会绕过编译器的类型检查,从而可能导致程序出现难以发现的错误。
为什么 static_cast 会出问题?

static_cast 是一种安全的类型转换方式,它会进行编译时检查。对于不合法的类型转换,它会直接报错,避免出现运行时错误。

  • 比如,在上面的例子中,static_cast 不会允许将 const float* 转换为 T*,因为这违反了 C++ 的类型系统。
  • static_cast 可以在一些合法的情况下进行转换,例如从基类指针转换为派生类指针,但必须保证转换是合法的、符合类型层次结构的。
总结

C风格类型转换非常灵活,但也非常危险。它不会进行类型检查,可能导致未定义的行为。使用 static_castdynamic_castconst_castreinterpret_cast 等现代 C++ 类型转换工具,能够帮助我们更加安全地进行类型转换,确保类型匹配,并且提高代码的可读性和可维护性。
总之,建议尽量避免使用 C风格的类型转换,转而使用 C++ 提供的类型转换方式,避免潜在的风险。

http://www.dtcms.com/a/242863.html

相关文章:

  • S1240核心的连接关系和工作流程
  • 【动手学深度学习】3.2. 线性回归的从零开始实现
  • idea中黄色感叹号打开
  • 纯血Harmony NETX 5 打造趣味五子棋:(附源文件)
  • ArcGIS土地利用数据制备、分析及基于FLUS模型土地利用预测技术应用
  • 1.4 超级终端
  • gbase8s之message log rotate
  • 路径规划算法概论:从理论到实践
  • Python 基础语法(1)【 适合0基础 】
  • C# StringBuilder代码中预分配容量的作用
  • Java免费获取汇率工具实现
  • 【计算机组成原理 第5版】白、戴编著 第四章 指令系统 课后题总结
  • HarmonyOS 应用开发学习记录 - 从Windows开发者视角看鸿蒙开发
  • 搭建一个springColud 项目,从头开始,里面有订单,库存两个模块
  • 湖北理元理律师事务所服务方法论:债务规划中的倾听者价值
  • CentOS7下的Flink 集群部署
  • EtherCAT转CC-Link IE协议转换通讯助力库卡机器人与三菱PLC无缝对接
  • C++_红黑树
  • 《为什么 String 是 final 的?Java 字符串池机制全面解析》
  • 常用ADB命令
  • LeetCode 3442. Maximum Difference Between Even and Odd Frequency I
  • C# Serilog 日志
  • 全国空气质量监测站点数据分析:从原始数据到空间可视化
  • (八)深度循环神经网络:长序列建模、注意力机制与多模态融合
  • NY167NY171美光固态闪存NY176NY180
  • 交叉编译笔记
  • 开源高频电磁场与电磁波数值仿真软件
  • chrome插件中如何使用midscene.js
  • Cursor 工具项目构建指南:让 AI 审查 AI 生产的内容,确保生产的内容质量和提前发现问题
  • 64页|PPT|基于华为IPD与质量管理体系融合的研发质量管理:L1-L6分层架构驱动高效运营、标准化质量管理体系