woka

转载:const指针


const修饰指针和引用的用法,对于初学C++的人直是讳莫如深,不知所云.一旦你了解了其用法,一切便不值一哂了.下面我为读者一一释疑
   
大致说来其可分为三种情况: const修饰指针,const修饰引用,const修饰指针的引用
1.const
修饰指针 
   const
修饰指针又可分为三种情况
     const
修饰指针本身 
     const
修饰指针所指的变量(或对象
     const
修饰指针本身和指针所指的变量(或对象
(1).const
修饰指针本身 
    
这种情形下,指针本身为常量,不可改变,任何修改指针本身的行为都是非法的.例如
const int a = 1; 
const int b = 2; 
int i = 3; 
int j = 4; 
int* const pi = &i; //ok, pi
的类型为int* const , &i的类型为int* const 
int* const pi = &a; //error, pi
的类型为int* const, &a的类型为const int* const 
pi = &j; //error, 
指针是常量,不可变 
*pi = a; //ok, *pi
并没有限定是常量 ,可变 
    
由此看出,pi是常量,常量在初始化和赋值时,类型必须严格一致。也就是 
const
修饰指针本身时,=号两边的变量类型必须严格一致,否则不能匹配。 
(2).const
修饰指针指向的变量(或对象
    
此种情形下,通过间接引用指针不可改变变量的值,假设指针为p,*p不可变,下面以例子说明
const int *pi = &a; 
//or int const *pi = &a; 
//
两者毫无二致,不过BS喜欢前者,这也没什么技术上的优劣之分,也就是说const intint const可以互换.建议大家熟 
//
悉这两种形式,为简洁便,以后统统用前者
//ok, const
并不修饰指针本身,pi对赋值类型 
//
没要求 ,piint*型指针,所以所赋的必须是个地址值。 
const int *pi = &i; //ok ,pi
可赋值常量的地址,又可赋变量的地址 
const int *pi1 = &a; 
const int *pi = pi1; //ok 
*pi = j; //error,*pi 
不可变,不能更改指针的间接引用形式 
pi = &j; //ok,pi
可变 
pi = &b; //ok,pi
可变 
pi++; //ok 
--pi; //ok 
     
由此可见,pi是变量,可以赋值常量和变量的值,正如一个整型变量可赋整型数和整型变量一样.const修饰的不是指针本身,而是其间接引用,=号两边的类型不必严格匹配,如:const int* pi = &a;中,pi的类型为int*,&a的类型为const int* const,只要其中含有int* 就可以。又如:const int *pi = &j;中,pi的类型为int*,&j的类型为int* const,它向pi赋值并无大碍。 
(3)const
修饰指针本身和指针所指的变量(或对象
     
设有指针p,此种情形下,p*p都不可变.举例如下
const int* const pi = &a; 
//or int const* const pi = &a; 
//
const pi看作一体,就与(2)所述相同,只是要求pi必须为const,正如上所说,=号两边的类型不必严格匹配,但必须含有int*, &a的类型为const int* const,含有int*, 所以可以赋值。 
const int* const pi = &i; //ok, &i
类型为int* const,含有int*, 可赋值。 
const int *pi1 = &j; 
const int *const pi = pi1; //ok,  pi1
类型为int* 
pi = &b; //error, pi
不可变 
pi = &j; //error, pi
不可变 
*pi = b; //error, *pi
不可变 
*pi = j; //error, *pi
不可变 
pi++; //error 
pi不可变 
++i; //ok, =
号右边的变量(或对象)与所修饰的变量无关 
a--; //error, a
const 
     
这种情况,跟以上两种情形有联系。对const int* const pi = &a;我们可以这样看:const int*( const pi )= &a;(仅仅是表达需要),将const pi看作一体,就与上述分类(2)符合。只要含有int*便可

2.const
修饰引用 
    
这种情况比较简单,没有象修饰指针那样繁复,因为引用和引用对象是一体的,所以引用被const修饰只有一种类型。 
const
修饰引用,引用本身不可变,但引用的变量(或对象)可以改变.例如
const int& ri = a; //or int const & ri = a; ok, ri 
本身是常量,引用不区分类型 
const int& ri = i; //ok,
引用不区分类型 
ri++; //error, ri
为常量,不可变 
i++; //ok,=
右边的变量与引用无关 
ri=b; //error, ri
为常量 
i=j; //ok,=
右边的变量与引用无关 
int & const ri = i; //error,
不存在这种形式,没有意义 

3.const
修饰指针的引用 
    
引用只是个别名,这里与修饰指针类似,又分为三种情况
(1) 
     
先给个例子: 
const int *pi = &a; 
const int *&ri = pi; 
//or int const *&ri = pi; 
    
引用是引用对象的别名,正因为如此,ripi的别名,所以ri的类型必须与pi完全一致才行。这里pi的类型为int*,ri的类型也为int*,赋值可行。若const int *&ri = &a;正不正确?分析一下就知晓。ri类型为int*,&a的类型则为const int* const不匹配。 
const int *&ri = &i; //error,
类型不匹配,一为int*,一为int* const 
ri = &a; //ok 
ri = &i; //ok 
const int *pi1=&a; 
const int *pi2=&i; 
ri = pi1; //ok 
ri = pi2; //ok 
*ri = i; //error 
*ri = a; //error 
     
注意这与1-(2)的区别
(2) 
     
用例子说明
int *const &ri = &i; 
    
去掉ri左边的&,则为int *const ri,因为ri是别名,ri的类型应与赋值的数类型一致,ri类型为int *const,&iint *const,可以这么做
int *const &ri = pi; //error,
类型不合,一为int *const ,一为int * 
int *const &ri = &a; //error,
类型不合,一为int *const,一为const int* const 
(*ri)++; //ok 
i++; //ok 
ri = &i; //error 
   
这种情况下,ri为常量,不可更改
(3) 
     
用例子说明: 
const int* pi = &j; 
const int* const &ri = pi; //or int const * const &ri = pi;ok 
const int* const &ri = &i; //ok 
     ri
pi的别名,pi的类型应与ri一致。拿掉&,得const int* const ri ,把const  ri看作一体,很容易得出ri的类型信息,就象前面2-(3)所讨论的一样,可以得到赋给ri的只要含有类型int* 即可。pi的类型为int*,&i的类型为int* const ,可以这么做
const int * const &ri = &a; //ok 
ri++;  //error 
*ri = 6;  //error 
    
言尽于此,希望对初学者有所助益!

posted on 2009-10-14 19:22 woka 阅读(1088) 评论(0)  编辑 收藏 引用


只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   博问   Chat2DB   管理


<2009年10月>
27282930123
45678910
11121314151617
18192021222324
25262728293031
1234567

导航

统计

常用链接

留言簿(2)

随笔档案

搜索

最新评论

阅读排行榜

评论排行榜