C++类型转换运算符详解

目录
  • 老式显式类型转换
  • c++的显示类型转换
  • 为什么要有新的类型转换
  • 具体应该使用哪种转换
    • static_cast运算符
    • dynamic_cast运算符
    • const_cast运算符
    • reinterpret_cast运算符
  • 老式显式类型转换实际的转换方式
  • 总结

老式显式类型转换

(类型)表达式 c风格的强制类型转换

类型(表达式) 函数式的强制类型转换

1,最开始使用的是c风格的类型转换,但是为了能够使类型转换看起来更像是一个函数调用,因此引入了函数式的类型转换。函数式的类型转换能够像使用一个函数那样去进行转换,比c风格的要更好一些。

2,一般来说,建议以上两种类型转换方式不要使用,改为使用以下四种类型转换方式,如果你拒绝这个提议的话,那么建议使用函数式的类型转换方式。

c++的显示类型转换

旧式的类型转换存在很多问题,因此,在c++中引入了新的类型转换方式(当然,所谓新的也是几十年前的事情了)。

static_cast

dynamic_cast

const_cast

reinterpret_cast

为什么要有新的类型转换

这里主要存在两个问题

第一,老式类型转换没有具体区分到底要怎么转换,是上面四种类型中的哪一种,或者哪几种?好处是,比较方便,因为,使用老式类型转换,你无需具体区分到底使用的是哪一种,只要按照顺序依次试一试能不能转换就行了。坏处是,这种转换是比较危险的,或者可能是用户期望之外的。错误使用了类型转换,但是却没有被发现,依然成功的被转换了,这种行为会带来更大的危害。

第二,老式类型转换是比较难识别的,不管是c风格的也好,函数式的也好,都是只用一个括号就可以了。括号显然比起static_cast<>这样的形式难以识别的多。因此,在复杂一些的表达式中,如果使用了多次类型转换,我们甚至很难去把里面所有的类型转换全部找到,这也会造成很大的困扰。

具体应该使用哪种转换

以前没得选的时候,使用起来自然很容易,因为我们无需纠结使用哪种,直接用就行了。但是有了四种可供你选择以后,这个问题就变的复杂起来了。到底应该用哪一种呢?什么时候应该用什么呢?当然,如果你实在分不清楚的话,那么首先应该尝试一下static_cast

static_cast运算符

static_cast<类型>(表达式)

static_cast最常用的情况是,类型和要转换的内容之间可以发生隐式类型转换

#include <iostream>
using namespace std;
int main()
{
	double a = 5 / 2;
	double b = static_cast<double>(5) / 2;
	// a的结果为2,b的结果为2.5
	cout << "a: " << a << endl;
	cout << "b: " << b << endl;
}

如果通过static_cast去转换具有继承关系的类,那么向上或者向下转换都是可以被允许的,但是转换为无关的类型是不被允许的。

#include <iostream>
using namespace std;
int main()
{
	class A{};
	class B: public A{};
	class C{};
	A a;
	B b;
	C c;
	A * p1 = static_cast<A *>(&b);
	B * p2 = static_cast<B *>(&a);
    // 转换为无关类型是不被允许的
	// C * p3 = static_cast<C *>(&a);
}

dynamic_cast运算符

dynamic_cast<类型>(表达式)

dynamic_cast只用于在继承层级上进行转换,可转换的类型为指针类型或者引用类型。

如果转换的类型为指针类型,且转换失败,那么将返回空指针。

如果转换的类型为引用类型,且转换失败,那么会抛出std::bad_cast类型异常。

注意:static_cast也可以用于这种类型转换,但是一般来说在处理dynamic_cast能解决的问题时,使用dynamic_cast要好于static_cast。

const_cast运算符

const_cast<类型>(表达式)

const_cast被用于移除或者添加cv限定符,当然主要是用于const

如果你还不知道cv限定符,那么可以点击这里查看有关cv限定符的描述

#include <iostream>
using namespace std;
int main()
{
	int a = 100;
	const int * p1 = &a;
	// 该赋值肯定是不被允许的
	// *p1 = 200
	// 该定义也是不被允许的,const int *类型不被允许用于初始化int *
	// int * p2 = p1;
	// 使用const_cast类型转换,将const int *类型转换为int *类型
	int * p2 = const_cast<int *>(p1);
	*p2 = 200;
	cout << a << endl;
}

reinterpret_cast运算符

reinterpret_cast<类型>(表达式)

这是一种非常危险的转换方式,它的功能是对类型进行重新解释。也就是直接将表达式的内容,在底层层级上视为新类型。使用该运算符时应该非常谨慎小心。

老式显式类型转换实际的转换方式

如果你使用了c风格或者函数式的显式类型转换,那么将会依次做这样的尝试,直到遇到第一个转换成功的选项为止。

const_cast

static_cast

static_cast const_cast

reinterpret_cast

reinterpret_cast const_cast

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注我们的更多内容!

(0)

相关推荐

  • C++隐式类型转换运算符operator type()用法详解

    在阅读<<C++标准库>>的时候,在for_each()章节遇到下面代码, #include "algostuff.hpp" class MeanValue{ private: long num; long sum; public: MeanValue():num(0),sum(0){ } void operator() (int elem){ num++; sum += elem; } operator double(){ return static_cast

  • 深入解析C++中的动态类型转换与静态类型转换运算符

    dynamic_cast 运算符 将操作数 expression 转换成类型为type-id 的对象. 语法 dynamic_cast < type-id > ( expression ) 备注 type-id 必须是一个指针或引用到以前已定义的类类型的引用或"指向 void 的指针".如果 type-id 是指针,则expression 的类型必须是指针,如果 type-id 是引用,则为左值. 有关静态和动态强制转换之间区别的描述,以及各在什么情况下适合使用,请参见 s

  • 浅谈C++类型转化(运算符重载函数)和基本运算符重载(自增自减)

    类型转化(运算符重载函数) 用转换构造函数可以将一个指定类型的数据转换为类的对象.但是不能反过来将一个类的对象转换为一个其他类型的数据(例如将一个Complex类对象转换成double类型数据).在C++提供类型转换函数(type conversion function)来解决这个问题.类型转换函数的作用是将一个类的对象转换成另一类型的数据. 类型转换函数的一般形式为: operator 类型名( ){ 实现转换的语句 } 下面是简单实现.这时候,Base起了两方面的作用:类和数据类型.系统会在

  • C++类型转换运算符的实例详解

    C++类型转换运算符的实例详解 C++中有4个类型转换运算符,使装换过程更规范 dynamic_cast; const_cast; static_cast; reinterpret_cast; 一.dynamic_cast 该运算符我在之前的文章中已经介绍过了 http://www.jb51.net/article/123252.htm 总之,该运算符的语法如下: dynamic_cast < type-name> (expression) 如果转型失败则返回0,即空指针. 该运算符的用途是,

  • C++类型转换运算符详解

    目录 老式显式类型转换 c++的显示类型转换 为什么要有新的类型转换 具体应该使用哪种转换 static_cast运算符 dynamic_cast运算符 const_cast运算符 reinterpret_cast运算符 老式显式类型转换实际的转换方式 总结 老式显式类型转换 (类型)表达式 c风格的强制类型转换 类型(表达式) 函数式的强制类型转换 1,最开始使用的是c风格的类型转换,但是为了能够使类型转换看起来更像是一个函数调用,因此引入了函数式的类型转换.函数式的类型转换能够像使用一个函数

  • python中的decimal类型转换实例详解

    [Python标准库]decimal--定点数和浮点数的数学运算 作用:使用定点数和浮点数的小数运算.         Python 版本:2.4 及以后版本 decimal 模块实现了定点和浮点算术运算符,使用的是大多数人所熟悉的模型,而不是程序员熟悉的模型,即大多数计算机硬件实现的 IEEE 浮点数运算.Decimal 实例可以准确地表示任何数,对其上取整或下取整,还可以对有效数字个数加以限制. Decimal 小数值表示为 Decimal 类的实例.构造函数取一个整数或字符串作为参数.使用

  • 基于C++ bitset常用函数及运算符(详解)

    C++ bitset--高端压位卡常题必备STL ------------------------------------------------------------ 以下内容翻译自cplusplus.com,极大地锻炼了我的英语能力. bitset存储二进制数位. bitset就像一个bool类型的数组一样,但是有空间优化--bitset中的一个元素一般只占1 bit,相当于一个char元素所占空间的八分之一. bitset中的每个元素都能单独被访问,例如对于一个叫做foo的bitset,

  • JavaScript知识点总结(四)之逻辑OR运算符详解

    在JavaScript中,逻辑OR运算符用||表示 var bTrue = true; var bFalse = false; var bResult = bTrue || bFalse; 下面的真值表描述了逻辑AND运算符的行为: 在JavaScript中,0,"",false,null,undefined,NaN均表示false, 可以用如下的代码证明: <script type="text/javascript"> var bFalse = fal

  • PHP读书笔记_运算符详解

    什么是运算符 什么是运算符?运算符是告诉PHP做相关运算的标识符号.例如,你需要计算123乘以456等于多少,这时候就需要一个符号,告诉服务器,你需要做乘法运算. PHP中的运算符有哪些?PHP运算符一般分为算术运算符.赋值运算符.比较运算符.三元运算符.逻辑运算符.字符串连接运算符.错误控制运算符. PHP中的算术运算符 算术运算符主要是用于进行算术运算的,例如:加法运算.减法运算.乘法运算.除法运算.在PHP中的常用的算术运算符对应下表: PHP中的赋值运算符 PHP的赋值运算符有两种,分别

  • C++位运算符详解(异或运算符和移位运算符)

    什么是位运算 位运算符按二进制进行运算,这些运算符只能用于整数类型的操作.如:char,short,int,long 通过位运算符来获取高位值和低位值 int a=0x1234; int high,low; high = (a>>8) &0x00ff; low = a & 0x00ff; 左移运算符和右移运算符(<<和>>) 左移是将一个二进制数,移动若干位,右边空出的位置用0来填补,高位左移溢出应该舍弃该高位. 如:inta = 8, a = 0000

  • 为IObservable实现自己的运算符(详解)

    您可以通过为LINQ库不提供的操作添加新运算符,或者通过创建自己的标准查询运算符实现来提高可读性和性能,从而扩展Rx. 编写自定义版本的标准LINQ操作符在您希望使用内存中对象操作时以及当预期自定义不需要查询的全面视图时非常有用. 创建新操作符 LINQ提供了一整套操作符,涵盖了一组实体上的大多数可能的操作. 但是,您可能需要一个运算符为查询添加特定的语义,尤其是如果您可以在代码中多次重复使用该运算符. 许多现有的LINQ运算符实际上是使用其他基本LINQ运算符构建的. 例如,SelectMan

  • 对Python3 * 和 ** 运算符详解

    在 Python 中,* 和 ** 具有语法多义性,具体来说是有四类用法. 1. 算数运算 * 代表乘法 ** 代表乘方 >>> 2 * 5 10 >>> 2 ** 5 32 2. 函数形参 *args 和 **kwargs 主要用于函数定义. 你可以将不定数量的参数传递给一个函数.不定的意思是:预先并不知道, 函数使用者会传递多少个参数给你, 所以在这个场景下使用这两个关键字.其实并不是必须写成 *args 和 **kwargs. *(星号) 才是必须的. 你也可以写

  • HashMap源码中的位运算符&详解

    引言 最近在读HashMap源码的时候,发现在很多运算符替代常规运算符的现象.比如说用hash & (table.length-1) 来替代取模运算hash&(table.length):用if((e.hash & oldCap) == 0)判断扩容后元素的位置等等. 1.取模运算符%底层原理 ​总所周知,位运算&直接对二进制进行运算:而对于取模运算符%:a % b 相当于 a - a / b * b,底层实际上是除法器,究其根源也是由底层的减法和加法共同完成.所以其运行效

  • 直观详细的typescript隐式类型转换图文详解

    正文 1.unknown是所有类型的父类型,其他类型都可以赋值给 unknown let a: undefined = undefined; let b: null = null; let x2: unknown; x2 = a; //正确 x2 = b; //正确 2.never 是任何类型的子类型,可以赋给任何类型 let a: undefined = undefined; let b: null = null; function err(): never { // OK throw new

随机推荐