【C++】类型转换方法

本篇博客让我们来见识一下C++中新增的类型转换方法

1.C语言中类型转换

在C语言中,类型转换有下面两种形式

  • 隐式类型转换
  • 显示强制类型转换int a=(int)'c'

这两种方式想必各位都很熟悉了,但隐式类型转换在一些场景里面会出现问题

void insert(size_t pos,char c)
{
	int end=10;
	while(end>=pos)
	{
		end--;
	}
	//...
}

上面的代码中,end是int类型,当进行比较的时候,end会-1直到小于pos

如果pos=0,问题就来了!

隐式类型转换会让end在比较的时候被转换为size_t无符号整型,而在无符号整型中,-1是一个非常大的正数,从而导致这个函数进入死循环!


  • 隐式类型转换可能会丢失数据的精度
  • 显示类型转换的写法都一样,导致不能很好的区分情况

C++委员会也是认识到了这里的问题,当产生隐式类型转换的时候,难以跟踪错误的来源,于是开发了下面的新的类型转换方式

不过,因为C++兼容C语言,所以C中的转换方式依旧支持


2.C++中的强制类型转换

C++中新增了下面四种明明的强制类型转换操作符

static_cast、reinterpret_cast、const_cast、dynamic_cast

2.1 static_cast

这个关键字是用于相近类型之间的转换的,比如double和int,char和int之间

	double d = 11.4;
	int a1 = static_cast<int>(d);//相近类型的转换
	char ch = 'a';
	int a2 = static_cast<int>(ch);
	cout << a1 << endl;
	cout << a2 << endl;

image-20221020162052557

2.2 reinterpret_cast

这个关键字用于不相近类型之间的转换,比如指针转成int

	int* p = &a1;
	//int x = static_cast<int>(p);//报错:类型转换无效
	int x = reinterpret_cast<int>(p);//非相近类型中的转换
	cout << x << endl;

打印的结果如下(每次运行都不一样)

2029408

2.3 const_cast

如同其名,这个关键字的作用是取消一个变量的const属性

const int c1 = 3;//这里定义的变量是在栈上的,可以间接修改
int* ptr1 = const_cast<int*>(&c1);//取消const权限
*ptr1 = 4;
cout << c1 << endl;//修改了地址之后没有变化
cout << *ptr1 << endl;

image-20221020162259828

欸,为什么我们取地址之后,修改为4了,变量c1本身不会变化呢?

这是因为编译器做了一些优化,把c1放到了某个地方,取的时候并没有直接去内存里面取

volatile关键字

这里我们可以使用volatile关键字修饰变量,要求每一次都必须要去内存中取

//volatile关键字,每次访问c都去内存中取,屏蔽编译器优化
volatile const int c2 = 3;
int* ptr2 = const_cast<int*>(&c2);//取消const权限
*ptr2 = 4;
cout << "volatitle:  ";
cout << c2 << endl;
cout << *ptr2 << endl;

image-20221020162443380

2.4 dynamic_cast

该关键字是用于继承中,将一个父类的指针/引用转换为子类对象的指针/引用

之前学习继承的时候,我们了解过

  • 向上转型:父类的指针、引用可以直接指向子类对象的指针/引用(这是一个赋值兼容的规则,不需要进行转换)
  • 向下转型:反过来之后,可以直接赋值吗?不够安全

dynamic_cast的作用就是判断一个父类指针指向的是不是他的子类

  • 如果是,能够成功转换
  • 不能则返回0

这个关键字最大的作用,便是可以帮我们判断这个父类指针/引用指向的是否为一个子类对象

class A
{
public:
	virtual void f() {}
};
class B : public A
{};

void fun(A* pa)
{
	// dynamic_cast会先检查是否能转换成功,能成功则转换,不能则返回0
	B* pb1 = static_cast<B*>(pa);
	B* pb2 = dynamic_cast<B*>(pa);
	cout << "pb1:" << pb1 << endl;
	if (pb2)
	{
		cout << "转换成功!pb2:" << pb2 << endl;
	}
	else
	{
		cout << "转换失败!pb2:" << pb2 << endl;
	}
	
}

void test2()
{
	A a;
	B b;
	fun(&a);
	fun(&b);
}

image-20221020163245904

这里有个要求,那便是父类中必须要有虚函数(多态),否则无法成功转换

image-20221020163349806

3.C++强制类型转换的作用

C++希望我们规范强制类型转换的情景,针对性的调用不同的关键字

但是由于它没有强制,在实际情况中用的反而不多

不过需要注意的是,强制类型转换会关闭/挂起正常的类型检查,在强制类型转换之前,我们要仔细检查是否还有别的方法来达到目的。最好是避免使用强制类型转换!

4.RTTI

之前学习智能指针的时候,我们学过一种思路叫RAII

这里的RTTI全称为Run-time Type identification,即运行时类型识别

C++通过下面几种方式来支持RTTI:

  • typeid
  • decltype
  • dyanmic_cast

这个概念只需要了解即可!

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

慕雪华年

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值