C++的类型转换

原创 2015年11月17日 16:49:25

类型转换名称和语法

C风格的强制类型转换(Type Cast)很简单,不管什么类型的转换统统是:

TYPE b = (TYPE)a   

C++风格的类型转换提供了4种类型转换操作符来应对不同场合的应用。

    static_cast                静态类型转换。如int转换成char

                   reinterpreter_cast  重新解释类型

           dynamic_cast           命名上理解是动态类型转换。如子类和父类之间的多态类型转换。

            const_cast            字面上理解就是去const属性。

4种类型转换的格式:

         TYPE B = static_cast<TYPE> (a) 


类型转换一般性介绍

         1static_cast<>()  静态类型转换,编译的时c++编译器会做类型检查

基本类型能转换 但是不能转换指针类型

         2)若不同类型之间,进行强制类型转换,用reinterpret_cast<>() 进行重新解释

         3)一般性结论:

C语言中  能隐式类型转换的,在c++中可用 static_cast<>()进行类型转换。因C++编译器在编译检查一般都能通过;

C语言中不能隐式类型转换的,在c++中可以用 reinterpret_cast<>() 进行强行类型 解释。总结:static_cast<>()reinterpret_cast<>() 基本上把C语言中的 强制类型转换给覆盖

reinterpret_cast<>()很难保证移植性。

         4dynamic_cast<>(),动态类型转换,安全的基类和子类之间转换;运行时类型检查

         5const_cast<>(),去除变量的只读属性      



static_cast用法和reinterpret_cast用法


void main()
{
         double dPi = 3.1415926;
 
         //1静态的类型转换:  在编译的时 进行基本类型的转换 能替代c风格的类型转换 可以进行一部分检查
         int num1 = static_cast<int> (dPi); //c++的新式的类型转换运算符 
         int num2 = (int)dPi;                              //c语言的 旧式类型转换
         int num3 = dPi;                                                        //隐士类型转换
         cout << "num1:" << num1 << " num2:" << num2 << " num3:" << num3 << endl;
 
 
         char *p1 = "hello wangbaoming " ;
         int *p2 = NULL;
         p2 = (int *)p1;
 
         //2 基本类型能转换 但是不能转换指针类型
         //p2 = static_cast<int *> (p1); //“static_cast”: 无法从“char *”转换为“int *”
 
         //3 可以使用  reinterpret_cast 进行重新解释
         p2 = reinterpret_cast<int *> (p1);
         cout << "p1 " << p1 << endl;
         cout << "p2 " << p2 << endl;
 
         //4 一般性的结论:      c语言中  能隐式类型转换的 在c++中可以用 static_cast<>()进行类型转换  //C++编译器在编译检查一般都能通过
         //c语言中不能隐式类型转换的,在c++中可以用 reinterpret_cast<>() 进行强行类型 解释
        
         system("pause");
         return ;
}

dynamic_cast用法和reinterpret_cast用法

class Animal
{
public:
         virtual void  cry() = 0;
};
 
class Dog : public Animal
{
public:
         virtual void  cry()
         {
                   cout << "wangwang " << endl;
         }
 
         void doSwim() 
         {
                   cout << "我要狗爬" << endl;
         }
};
 
 
class Cat : public Animal
{
public:
         virtual void  cry()
         {
                   cout << "miaomiao " << endl;
         }
         void doTree() 
         {
                   cout << "我要爬树" << endl;
         }
 
};
 
class Book
{
public:
         void printP()
         {
                   cout << price << endl;
         }
 
private:
         int price;
 
};
 
void ObjPlay(Animal *base)
{
         base->cry();
         Dog *pDog = dynamic_cast<Dog *>(base);
         if (pDog != NULL)
         {
                   pDog->cry();
                   pDog->doSwim();
         }
 
         Cat *pCat = dynamic_cast<Cat *>(base);
         if (pCat != NULL)
         {
                   pCat->cry();
                   pCat->doTree();
         }
}
void main02()
{
         Animal *base = NULL;
 
         //1 可以把子类指针赋给 父类指针 但是反过来是不可以的 需要 如下转换
         //pdog = base; 
         Dog *pDog = static_cast<Dog *> (base);
 
         //2 把base转换成其他 非动物相关的 err
         //Book *book= static_cast<Book *> (base);
 
         //3  reinterpret_cast //可以强制类型转换
         Book *book2= reinterpret_cast<Book *> (base);
 
         //4 dynamic_cast用法
         ObjPlay(new Cat());
 
         system("pause");
}



const_cast用法

//典型用法 把形参的只读属性去掉
void Opbuf(const char *p)
{
         cout << p << endl;
         char *p2 = const_cast<char*>(p);
         p2[0] = 'b';
         cout << p << endl;
}
 
void main()
{
         const char *p1 = "11111111111";
 
         char *p2 = "22222222";
 
         char *p3 = const_cast<char *>(p1);
         char buf[100] = "aaaaaaaaaaaa";
 
         Opbuf(buf);
 
         //要保证指针所执行的内存空间能修改才行 若不能修改 还是会引起程序异常
         //Opbuf("dddddddddddsssssssssssssss");
 
         system("pause");
}



总结

结论1:程序员要清除的知道要转的变量,类型转换前是什么类型,类型转换后是什么类型。转换后有什么后果。

结论2:一般情况下,不建议进行类型转换;避免进行类型转换。






C++中的#,##,和"

想要灵活应用宏,离不开#和##。 " 在学习#和##之前,先来看一个关于"的例子: #include #include int main() { const char* p1 = ...
  • mzlogin
  • mzlogin
  • 2014年11月09日 22:47
  • 7591

C++中的::的作用

(1)作用域限定符,当在类体中直接定义函数时,不需要在函数名字的前面加上类名,但是在类体外实现函数定义的时候,必须加上类名并且加上作用域限定符。Student::Display(); (2)...
  • zhanghuaichao
  • zhanghuaichao
  • 2017年02月18日 21:44
  • 2261

浅论C++的复杂性

C++语言已经有了20多年的历史。作为一门影响广泛的编程语言,它所受到的关注和争论恐怕是任何一门其他的语言所不能比拟的。十几年前,Java等新生语言的出现曾导致“C++信任危机”,但最终C++以自身非...
  • K346K346
  • K346K346
  • 2015年12月05日 11:56
  • 1738

c/c++中指针的理解(初学者)

关于指针,大家肯定不陌生,一些初学者,想必会出现思绪混乱的情况,现在我就来帮大家缕一缕吧。(第一次写微博 有点小紧张) 大家应该了解一些计算机对内存的管理方式吧。操作系统会将内存单元进行编号,这些...
  • Allen_ww
  • Allen_ww
  • 2016年01月11日 19:41
  • 8788

没事写点啥(一)——C++扫雷

整天编学校的作业实在枯燥,不如……写点别的?
  • destinyson
  • destinyson
  • 2016年02月29日 14:17
  • 1762

C++笔试题目大全(笔试宝典)(不断完善中)

1.new 、 delete 、 malloc 、 free 关系 delete 会调用对象的析构函数 , 和 new 对应 free 只会释放内存, new 调用构造函数。 malloc 与 f...
  • Lina_ACM
  • Lina_ACM
  • 2016年06月07日 13:29
  • 21419

C/C++中“#”和“##”的作用和用法

C/C++中“#”和“##”的作用和用法!
  • fengbingchun
  • fengbingchun
  • 2015年03月08日 16:15
  • 3385

C++ 进程间的通讯(一):简单的有名管道实现

进程间的通讯(一):简单的有名管道实现 一 管道简介 命名管道(Named Pipe)是服务器进程和一个或多个客户进程之间通信的单向或双向管道。不同于匿名管道的是命名管道可以在不相关的进程...
  • u010797208
  • u010797208
  • 2014年11月28日 02:09
  • 6576

OpenCV中Mat的C++用法介绍

Mat       OpenCV 自 2001 年出现以来。在那些日子里库是围绕C接口构建的。在那些日子里,他们使用名为IplImage C 的结构在内存中存储图像。这是您将在大多数较旧的教程和...
  • wadefan7
  • wadefan7
  • 2015年03月05日 15:53
  • 4295

关于C和C++一些大神们的讨论

知乎上曾经有一个关于C和C++语言的讨论,弄的沸沸扬扬。余天升 开源社区一直都不怎么待见C++,自由软件基金会创始人Richard Stallman认为C++有语法歧义,这样子没有必要、非常琐碎还会...
  • wangshubo1989
  • wangshubo1989
  • 2016年02月23日 20:33
  • 2637
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:C++的类型转换
举报原因:
原因补充:

(最多只允许输入30个字)