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

C++中的算术转换、其他隐式类型转换和显示转换详解

C++中的类型转换(Type Conversion)是指将一个数据类型的值转换为另一个数据类型的过程,主要包括:


一、算术类型转换(Arithmetic Conversions)

算术类型转换通常发生在算术运算或比较中,称为**“标准转换”(standard conversion),遵循一定的提升规则(promotion rules)**。

转换规则(简化版):

  1. boolint
  2. 整型提升:charshortint
  3. 两个操作数不同类型时:
    • 若一个为浮点型,另一个为整型 → 整型转换为浮点型
    • 若一个为double,另一个为floatfloat 转为 double
    • 若一个为unsigned且它的值不能用int表示 → int转为unsigned
    • 否则,unsigned提升优先级高于signed

示例:

#include <iostream>
int main() {char c = 100;int i = 200;float f = 1.5f;double d = 3.14;auto result1 = c + i;   // char → int,结果是 intauto result2 = i + f;   // int → float,结果是 floatauto result3 = f + d;   // float → double,结果是 doublestd::cout << "result1: " << result1 << "\n"; // 300std::cout << "result2: " << result2 << "\n"; // 201.5std::cout << "result3: " << result3 << "\n"; // 4.64
}

二、其他隐式类型转换(Implicit Conversion)

这是 编译器自动完成 的转换,也叫类型提升(promotion)或类型协同(coercion)

常见的隐式转换:

类型转换方向示例
数值型更大/更精度的数值intdouble
指针类型派生类 → 基类Derived*Base*
空指针nullptr → 任何指针类型nullptr → int*
数组/函数 → 指针int arr[]int*

示例:

void printDouble(double d) {std::cout << "Double: " << d << "\n";
}class Base {};
class Derived : public Base {};int main() {int i = 42;printDouble(i); // int → double(隐式)Derived d;Base* bptr = &d; // Derived* → Base*(隐式)
}

三、显示类型转换(Explicit Conversion / Cast)

你可以显式地指示编译器进行类型转换,有四种标准C++风格的转换方式(推荐),以及一个C风格的转换(不推荐)。

四种C++显式转换:

  1. static_cast<T>(expr)
    ➤ 用于基本类型之间转换指针/引用向上转换等安全转换。

  2. dynamic_cast<T>(expr)
    ➤ 用于带有虚函数的多态类之间的安全运行时类型检查和向下转换

  3. const_cast<T>(expr)
    添加或去除const、volatile限定符,常用于函数参数处理。

  4. reinterpret_cast<T>(expr)
    不安全但允许的位级别转换,如指针与整数之间转换。

  5. (T)expr —— C风格强制转换
    ➤ 组合了上述所有功能,不安全不透明,不推荐使用


示例:

#include <iostream>
class Base { public: virtual ~Base() {} };
class Derived : public Base { public: void say() { std::cout << "Derived\n"; } };int main() {int i = 100;double d = static_cast<double>(i);  // int → doubleBase* b = new Derived();Derived* dptr = dynamic_cast<Derived*>(b);  // 安全向下转型if (dptr) dptr->say();const int ci = 42;int* p = const_cast<int*>(&ci);  // 去除 const 限定void* vptr = reinterpret_cast<void*>(p); // int* → void*int* iptr = reinterpret_cast<int*>(vptr); // void* → int*delete b;
}

四、总结对比表:

转换类型触发方式安全性用途/说明
算术转换自动安全数值运算中的标准提升
隐式转换自动通常安全函数调用/指针/基本类型提升
static_cast显式安全编译期可检查的转换,如 int→float,向上转型
dynamic_cast显式安全多态类型的运行时向下转型
const_cast显式有风险去除 const/volatile
reinterpret_cast显式不安全位级别转换,极端情况用
C风格强制转换显式不推荐混合多个C++转换,缺乏类型安全

五、综合例子

下面是一个C++ 示例工程,通过一个模拟图形系统的类结构,演示了**各种类型转换(隐式、算术、显示转换)**的组合使用。代码里有注释,方便理解每一种转换在什么时候触发、为什么安全或者不安全。

定义 shape.h图形类头文件内容如下:

#pragma once
#include <iostream>
#include <string>class Shape {
public:virtual ~Shape() {}virtual void draw() const;
};class Circle : public Shape {
public:Circle(double r);void draw() const override;double getRadius() const;
private:double radius;
};class Rectangle : public Shape {
public:Rectangle(int w, int h);void draw() const override;int area() const;
private:int width, height;
};// 模拟图形渲染系统的底层资源对象
struct RenderHandle {void* data;
};RenderHandle createHandle(int id);

shape.cpp 图形类实现如下:

#include "shape.h"void Shape::draw() const {std::cout << "Drawing Shape\n";
}Circle::Circle(double r) : radius(r) {}
void Circle::draw() const {std::cout << "Drawing Circle with radius: " << radius << "\n";
}
double Circle::getRadius() const {return radius;
}Rectangle::Rectangle(int w, int h) : width(w), height(h) {}
void Rectangle::draw() const {std::cout << "Drawing Rectangle of area: " << area() << "\n";
}
int Rectangle::area() const {return width * height;
}RenderHandle createHandle(int id) {RenderHandle h;h.data = reinterpret_cast<void*>(id);  // 显式 reinterpret_castreturn h;
}

main.cpp 主函数演示各种类型转换示例内容如下:

#include "shape.h"
#include <typeinfo>void printDouble(double d) {std::cout << "Double value: " << d << "\n";
}int main() {// 算术类型转换(int → double)int i = 42;double d = i; // 隐式转换printDouble(d); // 参数是 doublefloat f = static_cast<float>(i); // 显式 static_caststd::cout << "Float from int: " << f << "\n";// 基类 → 派生类的动态转换(dynamic_cast)Shape* s = new Circle(3.5);s->draw();if (Circle* c = dynamic_cast<Circle*>(s)) {std::cout << "Radius from casted circle: " << c->getRadius() << "\n";}// const_cast 演示const int ci = 100;int* modifiable = const_cast<int*>(&ci);*modifiable = 999; // 未定义行为,仅演示用std::cout << "Modified const int (unsafe!): " << *modifiable << "\n";// reinterpret_cast 演示int resourceID = 1234;RenderHandle h = createHandle(resourceID);int recoveredID = reinterpret_cast<int>(h.data);std::cout << "Recovered ID from RenderHandle: " << recoveredID << "\n";// 隐式派生 → 基类(安全)Rectangle rect(10, 20);Shape* s2 = &rect;s2->draw();  // 虚函数,多态行为delete s;return 0;
}

示例的演示内容如下:

场景类型转换涉及关键字/方式
int → double算术隐式转换自动
float ← int算术显式转换static_cast
Base* ← Derived*隐式向上转换自动
Derived* ← Base*安全运行时转换dynamic_cast
const int* ← int*添加/去除 constconst_cast
void* ←→ int指针/整数转换reinterpret_cast
Shape 多态行为虚函数机制隐式

相关文章:

  • 极验4滑块笔记:整理思路--填坑各种问题
  • Java--数组的应用
  • Linux随记(十七)
  • 辛格迪客户案例 | 浙江高跖医药委托生产质量管理协同(OWL MAH)项目
  • C语言实现堆(优先队列)详解
  • 【沉浸式求职学习day21】【常用类分享,完结!】
  • 使用 Vue Router 和 Vite 构建的自动路由生成系统
  • Python基础总结(九)之推导式
  • C# 封装教程
  • 【SF顺丰】顺丰开放平台API对接(注册、API测试篇)
  • 《一次静态 ObjectMapper 引发的 RocketMQ 消费异常排查》
  • 极刻云搜-专业的软件网址搜索引擎
  • Linux421用户、组
  • 移动端动态滑动拨盘选择器【Axure元件库】
  • CMake execute_process用法详解
  • HyperDefect-YOLO:基于超图计算的工业缺陷检测算法解析
  • cdq 系列 题解
  • arkTs:使用Refresh实现下拉刷新功能(含状态提示与动画控制)
  • 并发设计模式之双缓冲系统
  • 基于SpringBoot的心情疗愈平台-项目分享
  • 山西一国道塌陷致2死后续:地质雷达检测出10处道路病害
  • 中国旅游日主题月期间,东航将准备超51.9万套特惠机票
  • 上海农房翻建为何难?基层盼政策适度松动
  • 上海交大发布“AI十条”,鄂维南院士已任该校人工智能学院讲席教授
  • 圆桌|耐心资本对科技创新有何意义?天使投资最关注哪些要素?
  • 人民日报和音:书写周边命运共同体建设新篇章