强制类型转换(const_cast)

来源:https://www.cnblogs.com/Braveliu/p/3616953.html

强制类型转换

const_cast的作用

  1. 常量指针 被强转为 非常量指针,且仍然指向原来的对象;

  2. 常量引用 被强转为 非常量引用,且仍然指向原来的对象;

  3. 常量对象 被强转为 非常量对象,此过程赋值运算符构造了新的对象,新的对象去掉了源对象的部分const属性。

实列代码
代码如下:

#include <iostream>
using  namespace std;

const int xx = 50;

class A
{
public:
	int m_nNum;

public:
	A(int nValue = 100);
};

A::A(int nValue) : m_nNum(nValue)
{
}
// 1.指向常对象的指针,并不是该对象无法被修改,而是该指针没有修改对象的权限,就算该指针被重新赋值,其仍为指向常对象的指针,无法对新对象进行修改。
//
void TestFun()
{
	// 第一种情况: const修饰指针指向对象
	const A *pA = new A(200);
	// pA->m_nNum = 100;            // compile error ! pA指针指向的对象为常对象,其成员变量值为只读的。
	A* pAA = const_cast<A*>(pA); // 去掉pA指针的const属性,将其赋值给另一个指针pAA,pA仍是指向常对象的指针
	pAA->m_nNum = 199;           // pAA指针指向的对象为一般对象,其成员变量值可读写。
	cout << pA->m_nNum << endl;  // 199

	// 第二种情况: const修饰指针
	A *pB = new A();
	pA = pB;  // pA是指向常对象的指针,pA是可以被赋值的,但pA新指向的对象仍无法被修改
	A* const pC = new A(1);  // 常指针pC,该指针无法被赋值
	cout << pC->m_nNum << endl;  // 1
	A *pD = new A(2);
	//  pC = pD;                     // compile error ! pC指针变量被const修饰,其值是只读的。

		
	A*& pE = const_cast<A*&>(pC); // 去掉pC指针变量的const属性。再赋给指针引用变量
	pE = pD;  // 成功改变了常量指针pC的值
	cout << pC->m_nNum << endl;  // 2 
		

	A* pAS = const_cast<A*>(pC); // 去掉pC指针变量的const属性。再赋给一般指针变量
	pAS = pB;

	// 第三种情况:const修饰指针和指针对象
	const A* const pCC = new A(110);
	const A* pCC2 = const_cast<A*>(pCC);  // 去掉了指针无法修改的const
	//  pCC2->m_nNum = 119; //  error C3490: 由于正在通过常量对象访问“m_nNum”,因此无法对其进行修改
	pCC2 = NULL;
	A* const pCC3 = const_cast<A*>(pCC);  // 去掉了指向对象无法被修改的const
	pCC3->m_nNum = 119;
	//  pCC3 = NULL; error C3892: “pCC3”: 不能给常量赋值
	A* pCC4 = const_cast<A*>(pCC);  // 一次性去掉上述两种const
	pCC4->m_nNum = 120;
	pCC4 = NULL;

	// 第四种情况:const修饰对象,去const属性后赋给一般对象
	const A a;
	//  a.m_nNum = 101; // compile error ! 常对象具有只读属性。
	A b = const_cast<A&>(a);  // 赋值操作构造了新的对象,新对象去掉了元对象的const属性
	b.m_nNum = 101;
	cout << a.m_nNum << endl; // 100
	cout << b.m_nNum << endl; // 101

	// 第五种情况:const修饰对象,去const属性后赋给引用对象
	const A c;
	//  c.m_nNum = 101; // compile error ! 常对象具有只读属性。
	A& d = const_cast<A&>(c);  // 引用对象去掉了原对象的const属性
	d.m_nNum = 102;
	cout << c.m_nNum << endl; // 102
	cout << d.m_nNum << endl; // 102

	// 第六种情况:const修饰对象,对象地址(相当于常指针)去const属性后赋给指针
	const A e;
	//  e.m_nNum = 103; // compile error ! 常对象具有只读属性。
	A* pe = const_cast<A*>(&e);
	pe->m_nNum = 103;
	cout << e.m_nNum << endl; // 103
	cout << pe->m_nNum << endl; // 103

	// 第七种情况:const修饰局部变量
	// 此处需要添加volatile关键字,不加的话,编译器会自动优化xx为50,因为编译器认为常对象不可更改
	// 仅针对内置数据类型,自定义数据类型不会进行这种优化,不加volatile的结果为50,200,50
	const volatile int xx = 50;  
	int* yy = const_cast<int *>(&xx);  // 该指针指向原变量位置,值为50
	*yy = 200;  // 成功修改了常量对象
	cout << xx << endl; // 200
	cout << *yy << endl; // 200
	int aa = xx;
	cout << aa << endl; // 200,

	// 第八种情况:const修饰局部变量。去const属性后赋给一般变量
	const int xxx = 50;  
	int yyy = const_cast<int&>(xxx);  // 赋值操作,一样构造了新的对象
	yyy = 51;
	cout << xxx << endl; // 50
	cout << yyy << endl; // 51

	// 第九种情况:const修饰局部变量。去const属性后赋给引用变量
	const volatile int xxx2 = 50;  // 与第七中情况类似,不加volatile的结果为50,52
	int& yyy2 = const_cast<int&>(xxx2);  // 该引用指向原对象
	yyy2 = 52;  // 通过引用也成功修改了常对象的值
	cout << xxx2 << endl; // 52
	cout << yyy2 << endl; // 52
}

int main()
{
	TestFun();
	system("pause");
	return -1;
}

输出结果:

199
1
2
100
101
102
102
103
103
200
200
200
50
51
52
52
  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C++中有四种强制类型转换,分别是reinterpret_cast、const_cast、static_cast和dynamic_cast。 reinterpret_cast用于进行底层的重新解释转换,可以将一个指针或引用转换为其他类型的指针或引用,甚至可以将指针转换为整数类型。但是使用reinterpret_cast需要非常小心,因为它是一种非常危险的操作,可能会导致未定义的行为。 const_cast用于移除const属性,可以将const修饰的变量或指针转换为非const类型。这可以用于去除const限定符,从而修改被const修饰的变量的值。 static_cast用于进行常见的类型转换,可以将一个表达式转换为另一种类型,例如基本数据类型之间的转换、父类指针或引用转换为子类指针或引用、以及void指针和任意类型的指针之间的转换。但是需要注意,在进行父类到子类的转换时,只有当父类指针或引用实际上指向了一个子类对象时,才能进行成功的转换。 dynamic_cast用于在继承关系中进行安全的向下转型(downcasting)。它可以将一个父类指针或引用转换为子类指针或引用,同时会进行类型检查,确保转换是安全的。如果转换失败,dynamic_cast会返回一个空指针或抛出一个std::bad_cast异常。 这四种强制类型转换在不同的场景下有不同的应用,可以根据具体的需求选择合适的转换方式。但是需要注意,在使用这些强制类型转换时,一定要谨慎和慎重,确保转换的安全性和正确性。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [C++中的类型转换static_cast、dynamic_cast、const_cast和reinterpret_cast总结](https://download.csdn.net/download/weixin_38629976/12808232)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] - *2* [C++的四种类型转换reinterpret_cast/const_cast/static_cast /dynamic_cast](https://blog.csdn.net/salmonwilliam/article/details/113941785)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] - *3* [C++四种cast转换(const_cast、static_cast、dynamic_cast、reinpreter_cast)类型转换运算符](https://blog.csdn.net/Dontla/article/details/130792118)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 33.333333333333336%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值