C++类型转换关键字(*_cast)

 在C/C++语言中用 (type) value(在C++还可以采用type(value))来进行显式类型转换(explicit type conversion),常常又被称为强制转换(cast投射/铸模)。这种转换的正确性完全掌握在程序员手中,传统上强制转换往往被过度使用,成为C++程序犯错的一个主要根源。
为了减少强制转换的副作用,并且在查错时使程序员能够快速定位(总是最值得怀疑的)强制转换,在标准C++中新增加了4个关键字*_cast,用来提倡一种全新的C++显式转换语法:

*_cast <type-id> (expression)

l         static_cast(静态转换):用于明确定义良性和适度良性的转换,包括原来不需要采用强制转换的自动类型转换(包括无损的提升转换和可能丢失信息的窄化转换[narrowing conversion],对后者编译器一般会提出警告)。标准C++提倡对任何数据的类型转换(不论是自动的还是强制的),都采用新的*_cast显式类型转换方法。例如:

int i = 0x7fff;
long l;
float f;
char c;
// (1)典型的非强制转换(自动转换)
// 传统方式:
l = i;
f = i;
// 提倡的新方式:
l = static_cast<long>(i);
f = static_cast<float>(i);
// (2)窄化转换
// 传统方式:
// 会显示警告信息:
i = l; // 可能丢失数字
i = f; // 可能丢失信息
c = i; // 可能丢失数字
// 不显示警告信息(但仍然难定位):
i = (int)l;
i = (int)f;
c = (char)i;
// 提倡的新方式(不会显示警告信息,且易定位):
i = static_cast<int>(l);
i = static_cast<int>(f);
c = static_cast<char>(i);

l         const_cast(常量转换):可将(同数据类型的)常型(const)转换为非常型、将易变(volatile)型转换为非易变型。如果用于其他类型的转换,一般会产生一个编译错误。例如:

const int i = 0;
int *pi;
pi = &i; // 错误
pi = (int *)&i; // 被反对
pi = const_cast<int *>(&i); // 完美
long *pl = const_cast<long *>(&i); // 错误
volatile int k = 0;
int *pk = const_cast<int *>(&k); // 正确

l         dynamic_cast(动态转换):一种安全的向下类型转换(downcast)操作,用于在一个类继承层次上向下移动。

因为每个派生类的基类都只有一个,而且派生类本身又包含了几乎所有的基类信息(private型的除外),所以向上的类型转换(upcast)总是唯一的和比较安全的。
而一个基类往往有多个派生类,而且派生类中一般会在基类的基础上添加了一些特有的数据和操作,所以向下的类型转换总是多态的和不太安全的。
dynamic_cast提供了一种安全的向下类型转换操作,只有当类型转换是正确的并且转换取的成功,返回值才是所需要的指针;否则它将返回0(空指针NULL),表示不是正确的类型。
例如:
class Pet {……};
class Dog : public Pet {……};
class Cat : public Pet {……};
……
Pet *pPet = new Cat; // 向上的类型转换
Dog *pDog = dynamic_cast<Dog *>(pPet); // 类型错误,返回0(NULL)
Cat *pCat = dynamic_cast<Cat *>(pPet); // 类型正确,返回指针
Cat *pCat = static_cast<Cat *>(pPet); // 正确,减少运行时的开销
注意:dynamic_cast虽然安全,但是运行时需要一定开销,因此不提倡大量使用这种转换。如果你已经能够确认转换的正确性,则可以采用前面介绍过的(无运行时开销的)static_cast转换。只有当你实在无法确定转换是否正确时,才需要采用dynamic_cast转换。

l         reinterpret_cast(重解释转换):一种最有可能出问题的最不安全的类型转换。只是在下面的情形,才需要使用这种类型转换:当需要使用时,所得到的东西已经不同了,为了使它能够用于原来的目的,必须再次把它转换回来。例如:

const int sz = 100; // 定义数组大小,标准C++提倡用常型变量(而不是常数或
// 符号常量宏)
struct X {int a[sz];}; // 只包含一个整数数组的结构
X x; // 定义结构变量,此时结构中的数组元素的值无意义(需要初始化)
int *px = reinterpret_cast<int *> (&x); // 为了初始化,先把结构转化为int数组
for (int *i = px; i < px + sz; i++) *i = 0; // 将每个数组元素的值初始化为0
print(reinterpret_cast<X *> (px)); // 重新转换成结构指针,以便使用
// 也可以直接使用原来的标识符x
// 此语句相当于print(&x);
使用reinterpret_cast通常是一种不明智且不方便的编程方式。但是在必须使用时,它也是非常有用的。
 
在这四种强制转换中, static_cast 最常用(目前还没有流行起来,但是被标准 C++ 着力提倡)、 dynamic_cast 最重要、 const_cast 也有用、而 reinterpret_cast 则很少被使用。

width="728" scrolling="no" height="90" frameborder="0" align="middle" src="http://download1.csdn.net/down3/20070601/01184120111.htm" marginheight="0" marginwidth="0">
  • 1
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
### 回答1: 在 C++ 中,const_cast 是一种用于转换 const 或 volatile 类型的指针或引用的关键字。它可以去掉指针或引用类型的常量性或易变性,但是需要注意,这种转换可能会导致未定义的行为。 下面是一个使用 const_cast 的简单示例: ```c++ void func(const int* ptr) { int* writable = const_cast<int*>(ptr); *writable = 100; // 通过 writable 修改指向的值 } int main() { int num = 10; const int* ptr = &num; func(ptr); // 调用 func 函数,修改 num 的值 return 0; } ``` 这个例子中,我们定义了一个 const int 类型的指针 ptr,它指向变量 num。在 func 函数中,我们使用 const_cast 去除了 ptr 指针的常量性,并将其赋值给了 int* 类型的 writable 指针。然后,我们通过 writable 指针修改了 num 的值。 需要注意的是,使用 const_cast 进行类型转换时需要非常小心,因为它可能会破坏原本的常量性或易变性,导致未定义的行为。因此,应该尽可能避免使用 const_cast,除非确实需要修改 const 或 volatile 对象。 ### 回答2: const_cast是C++中的一个关键字,用于将const属性从一个指针或引用中移除。它可以显式地将常量指针或引用转换为非常量指针或引用。 const_cast的使用场景可以分为两种情况。第一种情况是当我们需要修改一个以const声明的非指针或引用类型的变量时,使用const_cast可以去除其const属性。例如,当我们传递一个常量引用给一个函数,而这个函数需要修改该变量的值时,我们可以使用const_cast将其转为非常量引用,从而实现对变量的修改。 另一种情况是当我们需要修改一个以const声明的指针或引用类型的变量所指向的内容时,使用const_cast也可以实现。这通常发生在一个函数接受非常量指针或引用类型的参数,但在内部需要对参数所指向的内容进行修改时。在这种情况下,使用const_cast将常量指针或引用转换为非常量的形式,然后对其所指向的内容进行修改,可以达到我们的目的。 需要注意的是,虽然const_cast可以改变指针或引用的常量属性,但其本质上并不会真正去除对象的常量性。通过const_cast的修改操作可能会引发未定义行为或安全问题,因此使用const_cast时应当非常谨慎。我们应当遵循C++的常量性规则,并确保对常量对象的操作不会导致不可预期的结果。 总之,const_cast是C++中的一个关键字,它的主要作用是用于将const属性从指针或引用中移除,从而实现对常量对象的修改。但使用const_cast需要谨慎,遵循C++常量性规则,以确保程序的正确性和安全性。 ### 回答3: 关键字const_cast是C++中的一个强制类型转换符号,用于去除指针或引用变量的const或volatile属性。 在C++中,const修饰的变量表示其值是不可修改的,而volatile修饰的变量表示该变量可能会在任何时间被修改或读取,不受编译器对该变量的优化。 const_cast用于改变指针或引用的常量或易变性属性,使其能够修改被const修饰的变量。 const_cast的使用方式如下: 1. 将指针或引用从const转换为非const:const_cast<T*>(ptr)。其中T为指针所指向的类型,ptr为被const修饰的指针。 2. 将指针或引用从volatile转换为非volatile:const_cast<T volatile*>(ptr)。其中T为指针所指向的类型,ptr为被volatile修饰的指针。 需要注意的是,const_cast只能用于去除指针或引用变量的常量属性,而不能改变实际的变量本身,否则会导致未定义的行为。另外,使用const_cast去除const或volatile属性应该是谨慎的行为,只有在确保被修改的变量实际上不会被修改的情况下才能使用。 综上所述,const_cast关键字C++中用于去除指针或引用变量的const或volatile属性,使其能够修改被const修饰的变量。但需要注意使用时的谨慎和合理性。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值