C++的四种强制类型转换为:static_cast、const_cast、reinterpret_cast和dynamic_cast

类型转换的一般形式:cast-name(expression);

static_cast

任何具有明确定义的类型转换,只要不包含底层const,都可以使用static_cast;

double slope = static_cast(j) / i;

注:
  顶层const:表示指针本身是个常量。如:int *const p;
  底层const:表示指针所指的对象是一个常量。如:int const *p;

const_cast

该运算符只能改变运算对象的底层const。

#include<iostream>
using namespace std;
int main()
{
const char *pc=" HDU";
char *p=const_cast<char *>(pc); //正确, cout<<"hello"<<p<<endl;
return 0;
}

注:此处只能用const_cast,而不能用static_cast;

reinterpret_cast

通常为运算对象的位模式提供较低层次上的重新解释。
注:
  1、在指针之间转换,将一个类型的指针转换为另一个类型的指针,无关类型;
  2、将指针值转换为一个整型数,但不能用于非指针类型的转换。

dynamic_cast

只用于对象的指针和引用,不能用于内置的基本数据类型的强制转换。
使用dynamic_cast进行转换的,基类中一定要有虚函数,否则编译不通过。
运行时类型识别,用于将基类的指针或引用安全地转换成派生类的指针或引用。

对指针进行dynamic_cast,失败返回null,成功返回正常cast后的对象指针;
对引用进行dynamic_cast,失败抛出一个异常bad_cast,成功返回正常cast后的对象引用。

对于“向上转换”(即派生类指针或引用类型转换为其基类类型),无论是指针还是引用向上转换都是安全地。
对于“向下转型”有两种情况:
  1、基类指针所指对象是派生类类型的,这种转换是安全的;
  2、基类指针所指对象为基类类型,在这种情况下dynamic_cast在运行时做检查,转换失败,返回结果为0;
  在引用上,dynamic_cast依旧是常用于“安全的向下转型”。与指针一样,引用的向下转型也可以分为两种情况,与指针不同的是,并不存在空引用,所以引用的dynamic_cast检测失败时会抛出一个bad_cast异常。

#include <iostream>
#include <cstring>
using namespace std;
class A
{
public:
virtual void f()
{
cout << "hello" << endl;
}
}; class B: public A
{
public:
void f()
{
cout << "hello2" << endl;
} }; int main()
{
A* a1=new B;//a1是A类型的指针指向一个B类型的对象
A* a2=new A;//a2是A类型的指针指向一个A类型的对象
B* b; b=dynamic_cast<B*>(a1);
//结果为not null,向下转换成功,a1之前指向的就是B类型的对象,所以可以转换成B类型的指针。
if(b==NULL)
cout<<"null"<<endl;
else
cout<<"not null"<<endl; b=dynamic_cast<B*>(a2);//结果为null,向下转换失败
if(b==NULL)
cout<<"null"<<endl;
else
cout<<"not null"<<endl;
return 0;
}

总结

  基本类型转换用static_cast。
  去const属性用const_cast。
  不同类型的指针类型转换用reinterpreter_cast。
  多态类之间的类型转换用daynamic_cast。

C++ 4种强制类型转换

C++的四种强制类型转换为:static_cast、const_cast、reinterpret_cast和dynamic_cast

类型转换的一般形式:cast-name(expression);

static_cast

任何具有明确定义的类型转换,只要不包含底层const,都可以使用static_cast;

double slope = static_cast(j) / i;

注:
  顶层const:表示指针本身是个常量。如:int *const p;
  底层const:表示指针所指的对象是一个常量。如:int const *p;

const_cast

该运算符只能改变运算对象的底层const。

#include<iostream>
using namespace std;
int main()
{
const char *pc=" HDU";
char *p=const_cast<char *>(pc); //正确, cout<<"hello"<<p<<endl;
return 0;
}

注:此处只能用const_cast,而不能用static_cast;

reinterpret_cast

通常为运算对象的位模式提供较低层次上的重新解释。
注:
  1、在指针之间转换,将一个类型的指针转换为另一个类型的指针,无关类型;
  2、将指针值转换为一个整型数,但不能用于非指针类型的转换。

dynamic_cast

只用于对象的指针和引用,不能用于内置的基本数据类型的强制转换。
使用dynamic_cast进行转换的,基类中一定要有虚函数,否则编译不通过。
运行时类型识别,用于将基类的指针或引用安全地转换成派生类的指针或引用。

对指针进行dynamic_cast,失败返回null,成功返回正常cast后的对象指针;
对引用进行dynamic_cast,失败抛出一个异常bad_cast,成功返回正常cast后的对象引用。

对于“向上转换”(即派生类指针或引用类型转换为其基类类型),无论是指针还是引用向上转换都是安全地。
对于“向下转型”有两种情况:
  1、基类指针所指对象是派生类类型的,这种转换是安全的;
  2、基类指针所指对象为基类类型,在这种情况下dynamic_cast在运行时做检查,转换失败,返回结果为0;
  在引用上,dynamic_cast依旧是常用于“安全的向下转型”。与指针一样,引用的向下转型也可以分为两种情况,与指针不同的是,并不存在空引用,所以引用的dynamic_cast检测失败时会抛出一个bad_cast异常。

#include <iostream>
#include <cstring>
using namespace std;
class A
{
public:
virtual void f()
{
cout << "hello" << endl;
}
}; class B: public A
{
public:
void f()
{
cout << "hello2" << endl;
} }; int main()
{
A* a1=new B;//a1是A类型的指针指向一个B类型的对象
A* a2=new A;//a2是A类型的指针指向一个A类型的对象
B* b; b=dynamic_cast<B*>(a1);
//结果为not null,向下转换成功,a1之前指向的就是B类型的对象,所以可以转换成B类型的指针。
if(b==NULL)
cout<<"null"<<endl;
else
cout<<"not null"<<endl; b=dynamic_cast<B*>(a2);//结果为null,向下转换失败
if(b==NULL)
cout<<"null"<<endl;
else
cout<<"not null"<<endl;
return 0;
}

总结

  基本类型转换用static_cast。
  去const属性用const_cast。
  不同类型的指针类型转换用reinterpreter_cast。
  多态类之间的类型转换用daynamic_cast。

http://www.frankyang.cn/2017/05/10/c-4%E7%A7%8D%E5%BC%BA%E5%88%B6%E7%B1%BB%E5%9E%8B%E8%BD%AC%E6%8D%A2/

C++ 4种强制类型转换的更多相关文章

  1. C#中,三种强制类型转换的对比

    在C#中,我们可以看到三种强制类型转换,比如强制转换成有符号32位整型,可以找到下面三种方式: ① (int)()                ②Convert.ToInt32()          ...

  2. 四种强制类型转换的总结(const_cast、static_cast、dynamic_cast、reinterpreter_cast)

    四种强制类型转换的总结(const_cast.static_cast.dynamic_cast.reinterpreter_cast) 转载 2011年10月03日 23:59:05 标签: stru ...

  3. C++中的关键字用法--- 四种强制类型转换的总结

    四种强制类型转换的总结(const_cast.static_cast.dynamic_cast.reinterpreter_cast 1. C风格的强制类型转换(Type Cast)很简单,不管什么类 ...

  4. [转载]C++中四种强制类型转换方式

    C++中四种强制类型转换方式 原文地址:http://www.cnblogs.com/home123/p/6763967.html 类型转换有c风格的,当然还有c++风格的.c风格的转换的格式很简单( ...

  5. 【C++】 四种强制类型转换(static_cast 与 dynamic_cast 的区别!)

    强制类型转换 1. static_cast 2. dynamic_cast 3. const_cast 4. reinterpret_cast 5. 为什么要需要四种类型转换? 1. static_c ...

  6. C++的几种强制类型转换

    有时我们希望显式地将对象强制类型转换成另外一种类型.例如,如果想在下面的代码中执行浮点数除法: int i, j; double slope = i / j; 就要使用某种方法将i和/或j显式地转换成 ...

  7. C++开发必看 四种强制类型转换的总结 [转]

    一.C风格的强制类型转换(Type Cast)很简单,不管什么类型的转换统统是:     TYPE b = (TYPE)a 二.C++风格的类型转换提供了4种类型转换操作符来应对不同场合的应用. co ...

  8. C++四种强制类型转换关键字

    C语言使用强制类型转换(Type Cast)很简单,不管什么类型的转换,形式都如下: TYPE b = (TYPE)a; c++提供了4种类型转换操作符来应对不同场合的应用. const_cast   ...

  9. C++开发必看 四种强制类型转换的总结

    C风格的强制类型转换(Type Cast)很简单,不管什么类型的转换统统是: TYPE b = (TYPE)a C++风格的类型转换提供了4种类型转换操作符来应对不同场合的应用. const_cast ...

随机推荐

  1. Java自定义注解Annotation详解

    注解相当于一种标记,在程序中加了注解就等于为程序打上了某种标记,没加,则等于没有某种标记,以后,javac编译器,开发工具和其他程序可以用反射来了解你的类及各种元素上有无何种标记,看你有什么标记,就去 ...

  2. Java笔记13:统计文件中每个字符出现的次数

    一.代码实现 import java.io.*; import java.util.*; /** 功能:统计文件中每个字符出现的次数 思路: 1.定义字符读取(缓冲)流 2.循环读取文件里的字符,用一 ...

  3. IIS漏洞过滤

    IIS漏洞报告会提示网站HEAD包含Server版本信息导致版本泄漏,看起来不是大问题的漏洞却被分到中危或高危的行列中, 因为攻击者可能使用被披露信息获取特定版本发现的安全漏洞以及利用程序. 下面提供 ...

  4. 【架构】Twitter高性能RPC框架Finagle介绍

    Twitter的RPC框架Finagle简介 Finagle是Twitter基于Netty开发的支持容错的.协议无关的RPC框架,该框架支撑了Twitter的核心服务.来自Twitter的软件工程师J ...

  5. (转)mysql的单向复制

    mysql的单向复制操作很简单:大概只需要二十分钟看完这篇文章就能搞定了.http://11837782.blog.51cto.com/11827782/1885967 为了提高主从服务器的健壮性,我 ...

  6. ispy 编译笔记

    xcopy "$(ProjectDir)dlls\$(PlatformName)\*.*" "$(ProjectDir)$(OutDir)" /Y if NOT ...

  7. P2P终结者和反P2P终结者如何使用

    1 安装软件并运行,首先扫描网络,第一台控制机就是自己,你可以查看IP,和命令提示符下的IP吻合. 2 点击高级选项,指定本机网络环境和网卡 3 控制规则设置,首先设置全局限速模板,其他的差不多. 4 ...

  8. MySQL外键及级联删除 && 表的存储引擎与创建索引 && 删除数据库和表

    Messages表: mysql>create table Messages( ->message_id int auto_increment primary key, ->user ...

  9. 如何免密码直接登陆win7

    打开“运行”输入NETPLWIZ,并打开搜索到的程序将“要使用本机,用户必须输入用户名和密码”的勾去掉点 确定在弹出的对话框里填上你想要 自动登陆的 用户名和密码

  10. ios开发杂项(基础性介绍等)

    IOS Xcode开发中的文件后缀名区别m,mm,cpp,h .h :头文件.头文件包含类,类型,函数和常数的声明. .m :源代码文件.这是典型的源代码文件扩展名,可以包含Objective-C和C ...