C++指针之间的赋值与转换规则总结

Note:以下结论不适用于类的成员函数指针,关于类的成员函数指针会单独讨论。

一、任何类型的指针变量均可直接赋值给const void *

  任何类型的非const指针变量均可直接赋值给void * 

  const指针变量不可直接赋值给void *,除非通过强制类型转换

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
class A
{
};
 
typedef int (*pFun)(string); 
//函数指针
 
int *pInt;
const int *pInt_c;
char *pChar;
const char *pChar_c;
double *pDouble;
const double *pDouble_c;
A
*pA;         
//自定义类型指针
const A
*pA_c;
pFun
pf;   
//函数指针
 
void*
pVoid;
const void*
pVoid_c;
 
//
1.任何类型的指针变量均可直接赋值给const void *
pVoid_c
= pInt;        
//ok
pVoid_c
= pInt_c;      
//ok
pVoid_c
= pChar;       
//ok
pVoid_c
= pChar_c;     
//ok
pVoid_c
= pDouble;     
//ok
pVoid_c
= pDouble_c;   
//ok
pVoid_c
= pA;          
//ok
pVoid_c
= pA_c;        
//ok
pVoid_c
= pf;          
//ok
 
//
2.任何类型的非const指针变量均可直接赋值给void *
pVoid
= pInt;          
//ok
pVoid
= pChar;         
//ok
pVoid
= pDouble;       
//ok
pVoid
= pA;            
//ok
pVoid
= pf;            
//ok
 
//
3.const指针变量不可直接赋值给void *,除非通过强制类型转换
pVoid
= pInt_c;           
//error:
cannot convert from 'const int *' to 'void *'
pVoid
= pChar_c;          
//error:
cannot convert from 'const char *' to 'void *'
pVoid
= pDouble_c;        
//error:
cannot convert from 'const double *' to 'void *'
pVoid
= pA_c;             
//error:
cannot convert from 'const A *' to 'void *'
pVoid
= (
void*)pInt_c;    
//ok
pVoid
= (
void*)pChar_c;   
//ok
pVoid
= (
void*)pDouble_c; 
//ok
pVoid
= (
void*)pA_c;      
//ok

  

二、任意类型指针变量之间均可以强制类型转换,包括const与非const指针变量之间的强制类型转换。

1
2
3
4
5
6
7
8
9
pInt
= (
int*)pDouble;     
//ok
pInt
= (
int*)pf;          
//ok
pInt
= (
int*)pInt_c;      
//ok:由const指针变量转非const指针变量
pInt
= (
int*)pA_c;        
//ok:由const指针变量转非const指针变量
pA
= (A*)pA_c;            
//ok:由const指针变量转非const指针变量
pA
= (A*)pDouble;         
//ok
pA
= (A*)pf;              
//ok        
pf
= (pFun)pDouble;       
//ok
pf
= (pFun)pA;            
//ok

  

三、有继承关系的自定义类型之间:子类型指针变量可直接赋值给父类型指针变量

父类型指针变量不可直接赋值给子类型指针变量,除非通过强制类型转换

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
class A
{};
 
class B
:
public 
//
B继承自A
{};
 
class C
{};
 
A*
pA;
B*
pB;
C*
pC;
 
pA
= pB;       
//ok:
子类型指针变量可直接赋值给父类型指针变量
pB
= pA;       
//error:
父类型指针变量不可直接赋值给子类型指针变量,除非强制类型转换
                //以下适用规则二:
pA
= (A*)pC;   
//ok
pB
= (B*)pA;   
//ok
pB
= (B*)pC;   
//ok
pC
= (C*)pA;   
//ok
pC
= (C*)pB;   
//ok

  

补充:

1、对于类的成员函数指针,以上原则不适用。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
class A
{};
 
typedef void (A::*AFunPointer)(void);
typedef void (*FunPtr)(void);
 
void *
pVoid;
int *
pInt;
FunPtr
fp;
AFunPointer
afp;
 
pVoid
= afp;         
//error:
cannot convert from 'AFunPointer' to 'void *'
pInt
= (
int*)afp;    
//error:
'type cast' : cannot convert from 'AFunPointer' to 'int *'
fp
= (FunPtr)afp;    
//error:
'type cast' : cannot convert from 'AFunPointer' to 'FunPtr'
 
afp
= (AFunPointer)pInt;    
//error:
'type cast' : cannot convert from 'int *' to 'AFunPointer'
afp
= (AFunPointer)pVoid;   
//error:
'type cast' : cannot convert from 'void *' to 'AFunPointer'
afp
= (AFunPointer)fp;      
//error:
'type cast' : cannot convert from 'FunPtr' to 'AFunPointer'

我们可以这样理解:类的成员函数指针被限定在具体的某个类的作用域中了,他不能和域外的指针之间转换。

2、除去类的成员函数指针,虽然任意类型指针变量之间均可以强制类型转换,也即可以将const指针强转为非const指针。

但是应注意:如果将const指针强转为非const指针而用于原本const的对象,则产生未定义行为(C++语言未对此种情况进行规定)。如:

1
2
3
4
5
6
7
const int a
= 50;  
//
定义const变量及常量
const int*
p = &a; 
//
const指针变量p指向const变量a
int*
q = (
int*)p;  
//
const指针强转为非const指针,则非const指针变量q指向const变量a
*q
= 56;           
//
行为未定义,以下输出为VS2008下的输出结果
cout
<< a << endl;     
//输出:
50
cout
<< *p << endl;    
//输出:
56,很显然,p已经不指向a了
cout
<< *q << endl;    
//输出:
56,很显然,q已经不指向a了

3、关于一般函数指针的强制转换,以下当然也是OK的。

class A;
typedef void (*pFun1)(int, int);
typedef int (*pFun2)(A*, double); pFun1 pf1;
pFun2 pf2;
pf2 = (pFun2)pf1; // OK

C++指针之间的赋值与转换规则总结的更多相关文章

  1. C++:基类与派生类对象之间的赋值兼容关系

    4.5 基类与派生类对象之间的赋值兼容关系 在一定条件下,不同类型的数据之间可以进行类型转换,例如可以将整型数据赋给双精度型变量. 在赋值之前,先把整型数据转换为双精度型数据,然后再把它双精度型变量. ...

  2. C语言_指针变量的赋值与运算,很详细

    指针变量的赋值 指针变量同普通变量一样,使用之前不仅要定义说明, 而且必须赋予具体的值.未经赋值的指针变量不能使用, 否则将造成系统混乱,甚至死机.指针变量的赋值只能赋予地址, 决不能赋予任何其它数据 ...

  3. JS面向对象(2) -- this的使用,对象之间的赋值,for...in语句,delete使用,成员方法,json对象的使用,prototype的使用,原型继承与原型链

    相关链接: JS面向对象(1) -- 简介,入门,系统常用类,自定义类,constructor,typeof,instanceof,对象在内存中的表现形式 JS面向对象(2) -- this的使用,对 ...

  4. 使用 Attribute +反射 来对两个类之间动态赋值

    看同事使用的 一个ORM 框架 中 有这样一个功能  通过特性(附加属性)的功能来 实现的两个类对象之间动态赋值的 功能 觉得这个功能不错,但是同事使用的 ORM 并不是我使用的  Dapper  所 ...

  5. [C]关于函数指针参数的赋值

    问题 在有一次尝试用stat()函数获取文件属性的时候,发现如果直接声明一个指针,然后把这个指针作为参数传给函数,会导致函数执行失败,原代码: #include <sys/stat.h> ...

  6. C++ 指针悬挂和赋值操作符的重载,拷贝构造函数实现

    指针悬挂: 问题:使用new申请的内存内存空间无法访问,也无法释放. 原因:直接对指向new申请的存储空间的指针变量进行赋值修改 后果:失去了原来的地址,原来的空间无法访问也无法释放,造成内存泄漏 还 ...

  7. [错误记录_C] 还未给指针变量正确赋值的情况下,就使用它的值

    错误的代码: 错误的结果:  错误原因分析: 在使用(1) 将pB,pC的值赋给pA的lchild和rchild时: 还未给指针变量pB和pC赋值,现在pB和pC中存的是个垃圾值 Note: (2)- ...

  8. c语言中通过指针将数值赋值到制定内存地址

    1.一种直观的方法 假设现在需要往内存0x12ff7c地址上存入一个整型数0x100.我们怎么才能做到呢? 我们知道可以通过一个指针向其指向的内存地址写入数据,那么这里的内存地址0x12ff7c其本质 ...

  9. [转载]delete指针之后应该赋值NULL

    首先,C++标准规定:delete空指针是合法的,没有副作用.但是,delete p后,只是释放了指针指向的内存空间.p并不会自动被置为NULL,而且指针还在,同时还指向了之前的地址. 问题来了,对一 ...

随机推荐

  1. Hibernate课程 初探一对多映射5-1 课程总结

    1 单方一对多 xml one-to-many 配置 实体类   一方添加保存多方集合 2 单方多对一 xml many-to-one 配置 实体类   多方添加保存一方引用 3 常用属性 inver ...

  2. Axis Java调C# Webservice

    这是一个痛苦的过程,如果java对java的webservice可以说很方便,很简单,Axis,CXF等一系列框架生成客户端直接传参调用即可,但是异构语言就有点麻烦了,生成的客户端不好使......无 ...

  3. jQuery属性选择器中加变量

    $(function () { $('#bkhandle').on('click','#bkdel',function () { $.ajax( { url:"{% url 'bkdel' ...

  4. position的参考基准

    static(静态):position默认的样式:占据标准流的位置, 它会忽略top.bottom.left . right 的设置 relative(相对): 占据标准流的位置:可将其移至相对于其正 ...

  5. 画布canvas签名

    <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <m ...

  6. [C#]为什么Interface里的成员不能使用static修饰?

    首先引用MSDN里的原文 Interface members are automatically public, and they can't include any access modifiers ...

  7. Design Pattern ->Composite

    Layering & Contract Philosophy With additional indirection class CComponent { ; ; ; public: virt ...

  8. 【起航计划 012】2015 起航计划 Android APIDemo的魔鬼步伐 11 App->Activity->Save & Restore State onSaveInstanceState onRestoreInstanceState

    Save & Restore State与之前的例子Android ApiDemo示例解析(9):App->Activity->Persistent State 实现的UI类似,但 ...

  9. appium (三)执行过程

      转自http://blog.csdn.net/Yejianyun1/article/details/56012470 appium界面运行过程: 1.启动一个http服务器:127.0.0.1:4 ...

  10. intel Skylake平台安装WIN7

    目前针对IntelSkylake平台安装WIN7时USB接口失灵的问题,不少硬件厂商都推出了免费修改工具来集成XHCI USB控制器驱动,这其中技嘉提供了一款Windows USB Installat ...