C++ 类型转换

A. 隐式标准转换(Standard Conversion Sequence)

  1. 整型提升
cpp 复制代码
char c = 'A';
int  i = c;          // char → int 提升,无警告
std::cout << i;      // 65
  1. 整型 ↔ 浮点
cpp 复制代码
double d = 3;        // int → double
int    n = 3.14;     // double → int,截断为 3
  1. 数组退化(array-to-pointer)
cpp 复制代码
int a[4] = {1,2,3,4};
int* p = a;          // 数组名退化为 int*
  1. 指针转换
cpp 复制代码
int  x = 0;
int* p = &x;
void* pv = p;        // int* → void* 隐式

一句话规则:无需写任何关键字,编译器自动完成;只在"安全"场景下出现。


B. 用户定义转换(User-Defined Conversion)

  1. 转换构造函数
cpp 复制代码
struct Length {
    double m;
    Length(double x) : m(x) {}   // 转换构造
};
Length L = 42.0;     // double → Length,隐式
  1. 转换函数
cpp 复制代码
struct Money {
    double amount;
    explicit operator double() const { return amount; }
};
Money m{100};
double d = static_cast<double>(m);   // 必须显式

一句话规则:把"别的类型→本类型"写成构造函数;把"本类型→别的类型"写成 operator T()

坑点:explicit 能堵住隐式,但挡不住显式 cast。


C. 继承体系里的三种 cast

  1. 向上转型(Up-cast)
cpp 复制代码
struct Base { virtual ~Base() = default; };
struct Derived : Base {};

Derived d;
Base*   pb = &d;     // 隐式安全
  1. 向下转型(Down-cast)
cpp 复制代码
Base* pb = new Derived;
Derived* pd = dynamic_cast<Derived*>(pb);   // 安全,失败返回 nullptr
  1. 交叉转型(Cross-cast)
cpp 复制代码
struct A { virtual ~A() = default; };
struct B : virtual A {};
struct C : virtual A {};
struct D : B, C {};

D  obj;
B* pb = &obj;
C* pc = dynamic_cast<C*>(pb);  // B→C,成功

一句话规则:

  • 向上永远安全;
  • 向下/交叉用 dynamic_cast,否则 UB。

D. 对象切片(Slicing)

cpp 复制代码
struct Person { std::string name = "base"; };
struct Student : Person { int id = 42; };

Student s;
Person  p = s;      // 只拷贝 Person 部分,id 丢失
std::cout << p.name;   // base

一句话规则:把派生类对象 按值赋给/传参给 基类就会切片,改为指针/引用可避免。


E. 四种显式 cast

  1. static_cast(编译期检查)
cpp 复制代码
int   i = 10;
double d = static_cast<double>(i);
  1. dynamic_cast(运行期 RTTI)
cpp 复制代码
Base* pb = new Derived;
Derived* pd = dynamic_cast<Derived*>(pb);
  1. const_cast(仅改 cv 限定)
cpp 复制代码
const int ci = 5;
int* modifiable = const_cast<int*>(&ci);   // 注意:改值 UB
  1. reinterpret_cast(位级重解释)
cpp 复制代码
int  n = 0x1234;
int* p = &n;
std::uintptr_t v = reinterpret_cast<std::uintptr_t>(p);

一句话规则:

  • 数值/枚举/指针用 static_cast
  • 跨继承用 dynamic_cast
  • 去 const 用 const_cast
  • 位级解释用 reinterpret_cast

F. 模板与转换(enable_if / concepts)

cpp 复制代码
template<class T>
auto abs_val(T v)
    -> std::enable_if_t<std::is_signed_v<T>, T>
{
    return v < 0 ? -v : v;
}

C++20 Concepts 写法:

cpp 复制代码
template<std::signed_integral T>
T abs_val(T v) { return v < 0 ? -v : v; }

一句话规则:用 type traits 或 concepts 在编译期把"能否转换"写成约束。


G. C++20 bit_cast

cpp 复制代码
float f = 3.14f;
uint32_t u = std::bit_cast<uint32_t>(f);   // 与 memcpy 等效,安全

一句话规则:把 TriviallyCopyable 类型的位模式按字节搬到另一个同样大小的类型,零开销、无 UB。


使用速查表

需求 推荐 禁止
数值互转 static_cast C-style
派生→基类 隐式或 static_cast ---
基类→派生 dynamic_cast static_cast(可能 UB)
去掉 const const_cast reinterpret_cast
位级重解释 bit_cast / reinterpret_cast C-style
模板约束 enable_if / concepts ---

把示例直接拷进小文件即可独立编译运行。

相关推荐
Z1Jxxx几秒前
C++ P1151 子数整数
开发语言·c++·算法
User_芊芊君子几秒前
Python+Agent入门实战:0基础搭建可复用AI智能体
开发语言·人工智能·python
自信1504130575911 分钟前
重生之从0开始学习c++之类与对象(中)
c++·学习
ths51214 分钟前
测试开发python中正则表达式使用总结(二)
开发语言·python·算法
色空大师14 分钟前
【java打包方式详解】
java·开发语言·部署·打包·启动脚本·jar包分离
人道领域15 分钟前
2026年Java后端热点全景解析:从LTS革新到云原生跃迁
java·开发语言
heimeiyingwang15 分钟前
【架构实战】API接口防刷与限流策略
开发语言·python·架构
188号安全攻城狮17 分钟前
【前端基础知识】JavaScript 数组方法总结:从表格速查到分类详解
开发语言·前端·javascript·网络安全
不爱吃炸鸡柳17 分钟前
5道经典贪心算法题详解:从入门到进阶
开发语言·数据结构·c++·算法·贪心算法
智者知已应修善业26 分钟前
【51单片机1,左边4个LED灯先闪烁2次后,右边4个LED灯再闪烁2次:2,接着所用灯一起闪烁3次,接着重复步骤1,如此循环。】2023-5-19
c++·经验分享·笔记·算法·51单片机