返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >一起聊聊C++中的四种类型转换符
  • 608
分享到

一起聊聊C++中的四种类型转换符

2024-04-02 19:04:59 608人浏览 安东尼
摘要

目录一:背景二:理解四大运算符1. const_cast2. reinterpret_cast3. dynamic_cast3. static_cast一:背景 在玩 C 的时候,经

一:背景

在玩 C 的时候,经常会用 void* 来指向一段内存地址开端,然后再将其强转成尺度更小的 char* 或 int* 来丈量一段内存,参考如下代码:

int main()
{
	void* ptr = malloc(sizeof(int) * 10);

	int* int_ptr = (int*)ptr;
	char* char_ptr = (char*)ptr;
}

由于 C 的自由度比较大,想怎么玩就怎么玩,带来的弊端就是容易隐藏着一些不易发现的bug,归根到底还是程序员的功底不扎实,c++ 设计者觉得不能把程序员想的太厉害,应该要力所能及的帮助程序员避掉一些不必要的潜在 bug,并且还要尽最大努力的避免对性能有过多的伤害,所以就出现了 4 个强制类型转换运算符。

  • const_cast
  • reinterpret_cast
  • dynamic_cast
  • static_cast

既然 C++ 做了归类,必然就有其各自用途,接下来我们逐一和大家聊一下。

二:理解四大运算符

1. const_cast

这是四个运算符中最好理解的,玩过 C++ 的都知道,默认情况下是不能修改一个 const 变量,比如下面这样:

int main()
{
	const int i = 10;
	i = 12;
}

这段代码肯定是要报错的,那如果我一定要实现这个功能,如何做呢?这就需要用到 const_cast 去掉它的常量符号,然后对 i 进行操作即可,所以修改代码如下:

int main()
{
	const int i = 10;
	auto j = const_cast<int*>(&i);
	*(j) = 12;
}

2. reinterpret_cast

从名字上看就是一个 重新解释转换,很显然这个非常底层,如果大家玩过 windbg ,应该知道用 dt 命令可以将指定的内存地址按照某一个结构体丈量出来,比如说 C# 的 CLR 在触发 GC 时,会有 gc_mechanisms 结构,参考代码如下:

0:000> dt WKS::gc_mechanisms 0x7ffb6ba96e60
coreclr!WKS::gc_mechanisms
   +0x000 gc_index         : 1
   +0x008 condemned_generation : 0n0
   +0x00c promotion        : 0n0
   +0x010 compaction       : 0n1
   +0x014 loh_compaction   : 0n0
   +0x018 heap_expansion   : 0n0
   +0x01c concurrent       : 0
   +0x020 demotion         : 0n0
   +0x024 card_bundles     : 0n1
   +0x028 gen0_reduction_count : 0n0
   +0x02c should_lock_elevation : 0n0
   +0x030 elevation_locked_count : 0n0
   +0x034 elevation_reduced : 0n0
   +0x038 minimal_gc       : 0n0
   +0x03c reason           : 0 ( reason_alloc_soh )
   +0x040 pause_mode       : 1 ( pause_interactive )
   +0x044 found_finalizers : 0n0
   +0x048 background_p     : 0n0
   +0x04c b_state          : 0 ( bgc_not_in_process )
   +0x050 allocations_allowed : 0n1
   +0x054 stress_induced   : 0n0
   +0x058 entry_memory_load : 0
   +0x05c exit_memory_load : 0

其实 reinterpret_cast 大概也是干这个事的,参考代码如下:

typedef struct _Point {
	int x;
	int y;
} Point;

int main()
{
	Point point = { 10,11 };

	//内存地址
	void* ptr = &point;

	//根据内存地址 丈量出  Point
	Point* ptr_point = reinterpret_cast<Point*>(ptr);

	printf("x=%d", ptr_point->x);
}

从代码看,我直接根据 ptr 地址丈量出了 Point 结构,说实话这个和 C 玩法就比较类似了。

3. dynamic_cast

在多态场景下,有时候会遇到这样的一个问题,一个父类有多个子类,我现在手拥一个父类,我不知道能不能将它转换为其中一个子类,要试探一下看看,那怎么去试探呢? 类似 C# 中的 as 运算符,在 C++ 中就需要用 dynamic_cast 来做这件事情,参考如下:

//点
class Point {
public:
	Point(int x, int y) :x(x), y(y) {}
	virtual void show() {}
public:
	int x;
	int y;
};

//矩形
class Rectangle :public Point {
public:
	Rectangle(int x, int y, int w, int h) : Point(x, y), w(w), h(h) {}
public:
	int w;
	int h;
};

//三角形
class Triangle :public Point {
public:
	Triangle(int x, int y, int z) :Point(x, y), z(z) {}
public:
	int z;
};

int main()
{
	Point* p1 = new Rectangle(10, 20, 100, 200);
	Point* p2 = new Triangle(4, 5, 6);

	//将  p1 转成 子类 Triangle 会报错的
	Triangle* t1 = dynamic_cast<Triangle*>(p1);

	if (t1 == nullptr) {
		printf("p1 不能转成 Triangle");
	}
}

对,场景就是这个,p1 其实是 Rectangle 转上去的, 这时候你肯定是不能将它向下转成 Triangle , 问题就在这里,很多时候你并不知道此时的 p1 是哪一个子类。

接下来的一个问题是,C++ 并不像C# 有元数据,那它是如何鉴别呢? 其实这用了 RTTI 技术,哪里能看出来呢?哈哈,看汇编啦。

	Triangle* t1 = dynamic_cast<Triangle*>(p1);
00831D57  push        0  
00831D59  push        offset Triangle `RTTI Type Descriptor' (083C150h)  
00831D5E  push        offset Point `RTTI Type Descriptor' (083C138h)  
00831D63  push        0  
00831D65  mov         eax,dWord ptr [p1]  
00831D68  push        eax  
00831D69  call        ___RTDynamicCast (083104Bh)  
00831D6E  add         esp,14h  
00831D71  mov         dword ptr [t1],eax  

从汇编可以看到编译器这是带夹私货了,在底层偷偷的调用了一个 ___RTDynamicCast 函数在运行时帮忙检测的,根据 cdcel 调用协定,参数是从右到左,恢复成代码大概是这样。

___RTDynamicCast(&p1, 0, &Point, &Triangle,0)

3. static_cast

从名字上就能看出,这个强转具有 static 语义,也就是 编译阶段 就生成好了,具体安全不安全,它就不管了,就拿上面的例子,将 dynamic_cast 改成 static_cast 看看有什么微妙的变化。

int main()
{
	Point* p1 = new Rectangle(10, 20, 100, 200);
	Point* p2 = new Triangle(4, 5, 6);

	Triangle* t1 = static_cast<Triangle*>(p1);

	printf("x=%d, y=%d,z=%d", t1->x, t1->y, t1->z);
}

我们发现居然转成功了,而且 Triangle 的值也是莫名奇怪,直接取了 Rectangle 的前三个值,如果这是生产代码,肯定要挨批了。。。

接下来简单看下汇编代码:

	Triangle* t1 = static_cast<Triangle*>(p1);
00DF5B17  mov         eax,dword ptr [p1]  
00DF5B1A  mov         dword ptr [t1],eax  

	printf("x=%d, y=%d,z=%d", t1->x, t1->y, t1->z);
00DF5B1D  mov         eax,dword ptr [t1]  
00DF5B20  mov         ecx,dword ptr [eax+0Ch]  
00DF5B23  push        ecx  
00DF5B24  mov         edx,dword ptr [t1]  
00DF5B27  mov         eax,dword ptr [edx+8]  
00DF5B2A  push        eax  
00DF5B2B  mov         ecx,dword ptr [t1]  
00DF5B2E  mov         edx,dword ptr [ecx+4]  
00DF5B31  push        edx  
00DF5B32  push        offset string "x=%d, y=%d,z=%d" (0DF8C80h)  
00DF5B37  call        _printf (0DF145Bh)  
00DF5B3C  add         esp,10h

从代码中看,它其实就是将 p1 的首地址给了 t1,然后依次把copy偏移值 +4,+8,+0C, 除了转换这个,还可以做一些 int ,long ,double 之间的强转,当然也是一样,编译时汇编代码就已经生成好了。

到此这篇关于一起聊聊C++中的四种类型转换符 的文章就介绍到这了,更多相关C++类型转换符 内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: 一起聊聊C++中的四种类型转换符

本文链接: https://lsjlt.com/news/164342.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 一起聊聊C++中的四种类型转换符
    目录一:背景二:理解四大运算符1. const_cast2. reinterpret_cast3. dynamic_cast3. static_cast一:背景 在玩 C 的时候,经...
    99+
    2024-04-02
  • C++中的四种类型转换符是什么
    本篇内容介绍了“C++中的四种类型转换符是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!一:背景在玩 C 的时候,经常会用 v...
    99+
    2023-07-02
  • C++的四种类型转换
    目录一、隐式类型转换和显示类型转换二、C++的四种类型转换2.1 static_cast 相似转化2.2 reinterpret_cast 不同类型转化2.3 const_cast ...
    99+
    2023-05-14
    C++实现类型转换方法 C++实现类型转换
  • 一起聊聊PHP session共享的四种解决方案
    本文给大家介绍PHP session相关知识,下面带大家先从为什么会出现这种session共享的解决方案开始了解,希望对需要的朋友有所帮助~先了解一下为什么会出现这种session共享的解决方案?随着互联网公司的项目在微服务和分布式的环境下...
    99+
    2023-05-14
    php session
  • 详解C++中四种类型的转换
    目录const_caststatic_castreinterpret_castdynamic_cast代码C语言中我们使用 int a=(int) b;的方式强制转换 C++提供了四...
    99+
    2022-12-08
    C++类型转换 C++ 类型转换符 C++ 转换符
  • 一起聊聊C++中的智能指针
    目录一:背景二:关键词解析1. auto_ptr2. auto_ptr 多引用问题一:背景 我们知道 C++ 是手工管理内存的分配和释放,对应的操作符就是 new/dele...
    99+
    2024-04-02
  • 一文搞懂C++中的四种强制类型转换
    在了解c++的强制类形转换的时候,先看看在c语言中是怎么进行强制类形转换的。 C语言中的强制类形转换分为两种  隐式类型转换 显示类型转换 int ...
    99+
    2024-04-02
  • 一起详细聊聊C#中的Visitor模式
    目录写在前面模式演进举个例子使用了Tpye-Switch的版本尝试使用重载的版本单分派与双分派Visitor模式总结写在前面 Visitor模式在日常工作中出场比较少,如果统计大家不...
    99+
    2024-04-02
  • C++中的四种强制类型转换介绍
    这篇文章主要讲解了“C++中的四种强制类型转换介绍”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C++中的四种强制类型转换介绍”吧!在了解c++的强制类形转换的时候,先看看在c语言中是怎么进...
    99+
    2023-06-20
  • 一起聊聊Java中13种锁的实现方式
    目录1、悲观锁2、乐观锁3、分布式锁加锁4、可重入锁5、自旋锁6、独享锁7、共享锁8、读锁/写锁9、公平锁/非公平锁10、可中断锁/不可中断锁11、分段锁12、锁升级(无锁|偏向锁|...
    99+
    2022-11-13
    Java 实现锁 Java 锁
  • 一起聊聊C++中的特殊成员函数
    目录一:背景二:特殊成员函数1. 默认构造函数2. 析构函数3. 赋值构造函数4. 赋值运算符一:背景 在C#中要说类默认给我们定义的特殊成员函数,莫过于构造函数,但在 C++ 中这...
    99+
    2024-04-02
  • 聊聊Go语言中整数类型之间的转换操作
    Go语言中有多种整数类型,包括int、int8、int16、int32、int64、uint、uint8、uint16、uint32和uint64等。这些类型占用的字节数不同,在运算和存储时有各自的特点和限制。当我们需要对不同类型的整数进行...
    99+
    2023-05-14
  • C++强制类型转换的四种方式
    目录1 C++类型转换本质1.1 自动类型转换(隐式)1.2 强制类型转换(显式)1.3 类型转换的本质1.4 类型转换的安全性2 四种类型转换运算符2.1 C语言的强制类型转换与C...
    99+
    2024-04-02
  • C++中四种类型转换的方法是什么
    这篇文章主要介绍了C++中四种类型转换的方法是什么的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇C++中四种类型转换的方法是什么文章都会有所收获,下面我们一起来看看吧。一、隐式类型转换和显示类型转换当等号两边的...
    99+
    2023-07-05
  • 聊聊javascript中常见的一些转义字符
    JavaScript是一种基于文本的编程语言,因此它需要一种机制来处理特殊字符。这些特殊字符可以是控制字符,例如换行符和制表符,或者是一些需要转义的字符,例如引号和反斜杠。在JavaScript中,使用反斜杠(\)来指示特殊字符。这被称为转...
    99+
    2023-05-14
  • 聊聊Golang中的几种常用基本数据类型
    本篇文章带大家了解一下golang 的几种常用的基本数据类型,如整型,浮点型,字符,字符串,布尔型等,并介绍了一些常用的类型转换操作。Golang 是一种强类型语言,变量使用时需要进行强制类型定义,一旦某一个变量被定义,如果不经强制转换,那...
    99+
    2022-06-30
    Go Golang
  • C++实例讲解四种类型转换的使用
    目录C++类型转换C语言风格的转换C++风格的类型转换static_castreinterpret_castconst_castdynamic_cast小结C++类型转换 C语言风格...
    99+
    2024-04-02
  • 聊聊C语言中sizeof运算符的一个陷阱
    sizeof运算符通常用于获取变量或类型所占内存的大小(单位是字节) #include <stdio.h> struct D{ char a; int...
    99+
    2022-11-13
    c语言程序运算符sizeof sizeof c语言 C语言sizeof函数
  • C++11新特性之四种类型转换cast说明
    目录引言1、static_cast1.1 基本类型转换1.2 类的上行转换(安全)1.3 类的下行转换(不安全)2、const_cast2.1 改变常量属性3、dynamic_cas...
    99+
    2023-02-09
    c++11类型转换 c++11类型转换cast C++类型转换
  • C++中4种类型转换的方法分享
    目录1. C语言中的类型转换2. 为什么C++需要四种类型转换3. C++强制类型转换3.1 static_cast3.2 reinterpret_cast3.3 const_cas...
    99+
    2023-05-14
    C++实现类型转换方法 C++类型转换方法 C++类型转换
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作