const修饰指针和引用释疑(转载)

本文主要为C++初学者释疑const修饰指针和引用的用法。详细介绍了const修饰指针(包括修饰指针本身、指针所指变量、两者都修饰)、const修饰引用、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 int与int const可以互换.建议大家熟
//悉这两种形式,为简洁便,以后统统用前者.
//ok, const并不修饰指针本身,pi对赋值类型
//没要求 ,但pi是int*型指针,所以所赋的必须是个地址值。
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;
    引用是引用对象的别名,正因为如此,ri是pi的别名,所以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,&i为int *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
    言尽于此,希望对初学者有所助益!

http://dev.youkuaiyun.com/article/27/27622.shtm
<think>好的,用户之前问了关于C++const修饰类成员的问题,现在转向了const修饰指针的用法。需要结合用户提供的引用内容,整理出修饰指针本身指向数据的区别。 首先,用户引用了四个资料,其中提到了三种情况:常量指针指针常量两者都修饰。需要明确术语,可能用户对术语有混淆,比如“指针常量”“常量指针”容易搞反。根据引用中的例子,const int* p是常量指针指针指向的数据不可变;int* const p是指针常量,指针本身不可变。需要确保术语的正确使用,避免误导。 接下来,用户希望了解两者的区别用法,所以需要分点说明,可能用对比表格更清晰。要结合代码示例,展示不同情况下的修改权限。同时,注意用户提到的“修饰指针本身”修饰指针指向的数据”,需要明确这两种情况的语法效果。 另外,用户之前的回答中提到了const成员函数成员变量,可能需要区分指针作为成员时的const用法,但当前问题更关注指针变量本身的修饰,所以暂时不涉及类成员的情况。 还需要注意用户可能混淆的情况,比如const的位置不同导致的差异,以及如何记忆这些区别。比如,const在*左边还是右边,或者使用typedef时的特殊情况。引用中提到的情况需要涵盖,并给出对应的示例代码。 最后,总结两者的核心区别,强调应用场景,比如函数参数中const指针的使用,保护数据不被修改或者指针不被重新指向。可能用户在实际编程中遇到相关问题,比如传递只读数据或固定指针的情况,需要举例说明。</think>#### 一、核心区别对比 | 特征 | const修饰指针指向的数据 | const修饰指针本身 | 双重const修饰 | |--------------------|----------------------------------|--------------------------------|-----------------------------| | **语法形式** | `const T* p` 或 `T const* p` | `T* const p` | `const T* const p` | | **可变性** | 数据不可修改,指针可改变指向 | 数据可修改,指针不可改变指向 | 数据指针均不可修改 | | **内存保护级别** | 保护指向的数据 | 保护指针的指向地址 | 双重保护 | | **典型应用场景** | 函数参数传递只读数据 | 固定指针地址的硬件寄存器访问 | 完全不可变的指针配置 | --- #### 二、具体用法解析 ##### 1. **const修饰指针指向的数据(常量指针)** - **特征**:数据不可修改,指针可重新指向其他地址 - **代码示例**: ```cpp int a = 10, b = 20; const int* p1 = &a; // 等价于 int const* p1 // *p1 = 30; // 错误:禁止修改数据 p1 = &b; // 合法:允许修改指针指向 ``` ##### 2. **const修饰指针本身(指针常量)** - **特征**:指针地址固定,数据可通过指针修改 - **代码示例**: ```cpp int x = 5, y = 6; int* const p2 = &x; *p2 = 8; // 合法:允许修改数据 // p2 = &y; // 错误:禁止修改指针指向 ``` ##### 3. **双重const修饰(完全常量指针)** - **代码示例**: ```cpp const int* const p3 = &a; // *p3 = 50; // 错误:数据不可修改 // p3 = &b; // 错误:指针不可修改 ``` --- #### 三、深度解析与记忆技巧 1. **语法快速判断法** - `const`在`*`左侧:保护数据(例:`const int* p`) - `const`在`*`右侧:保护指针(例:`int* const p`)[^1][^2] 2. **typedef的特殊情况** ```cpp typedef int* IntPtr; const IntPtr p4 = &a; // 等价于 int* const p4 ``` 3. **函数参数中的典型应用** ```cpp // 保护数据不被修改 void printData(const char* str) { // str[0] = 'A'; // 编译错误 cout << str; } // 保护指针不被修改 void initBuffer(char* const buffer) { buffer[0] = '\0'; // 允许修改数据 // buffer = new char[100]; // 错误 } ``` --- #### 四、易错场景分析 1. **跨作用域修改风险** ```cpp const int* p = &a; int* q = const_cast<int*>(p); *q = 100; // 未定义行为,可能引发崩溃[^2] ``` 2. **复合类型中的const传播** ```cpp const int** pp1; int* p; // pp1 = &p; // 错误:const层级不匹配 ``` --- #### 五、应用场景建议 | 场景 | 推荐用法 | 优势说明 | |---------------------------|-------------------------|----------------------------| | 只读数据传递 | `const T*` | 防止意外数据修改 | | 硬件寄存器访问 | `T* const` | 固定物理地址,确保访问安全 | | 配置文件参数 | `const T* const` | 完全锁定指针数据 | [^1]: 指针常量与常量指针的核心区别在于`const`关键字的位置 [^2]: 强制修改`const`数据属于未定义行为,编译器可能不会报错但会导致运行时问题
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值