与引用一样,可以令指针指向常量或非常量。类似于常量引用,指向常量的指针不能用于改变其所指对象的值。
如果想要存放常量对象的地址,只能使用指向常量的指针:
const double pi = 3.14; // pi是一个常量,值不能发生改变
double *ptr = π // 错误, ptr只是一个普通的指针, 无法指向一个被const修饰的量
const double *cptr = π // 正确,cptr同样被const修饰,可以指向被const修饰的量
*cptr = 42; // 错误: 不能给*cptr赋值
通常情况下,指针的类型必须与其所指对象的类型一致,但是有两个例外。第一种例外情况是允许令一个指向常量的指针指向一个非常量对象:
double dval = 3.14; // dval 是一个双精度浮点数,它的值可以被改变
cptr = &dval; // 正确:但是不能通过cptr改变dval的值
和常量引用一样,指向常量的指针没有规定其所指的对象必须是一个常量。所谓指向常量的指针仅仅要求不能通过该指针改变对象的值,并未规定那个对象的值不能通过其他途径改变。
const指针
指针是对象而引用不是,就像其他对象一样,指针本身可以设置为常量。常量指针必须要初始化,而初始化完成后,它的值——地址,就不能再改变了。如果const之前用*修饰,则说明指针是一个常量,这样的书写隐含着更深层的含义,即不变的是指针本身的值而非指向的值:
int errNumb = 0;
int *const curErr = &errNumb; // curErr将一直指向errNumb
const double pi = 3.1415926;
const double *const pip = π // pip是一个指向常量对象的常量指针
而弄清楚这些声明的含义最有效的办法就是从右向左阅读。
此例中,离curErr最近的符号是const, 意味值curErr本身是一个常量对象,对象的类型由声明符的其余部分确定。声明符中的下一个符号是*,表示curErr是一个常量指针。
最后,该声明语句的基本数据类型部分确定了常量指针指向的是一个int对象。同样的,pip是一个常量指针,它指向的对象是一个双精度浮点型常量。
指针本身是一个常量并不意味着不能通过指针修改其所指对象的值,能否这样做取决于所指对象的类型。上例中pip是一个指向常量的常量指针,不论是pip所指对象还是pip自己的地址都不能改变,而curErr所指的是一个一般的非常量整数,那么可以用curErr去修改errNumb的值:
*pip = 2.72; // 错误:pip是一个指向常量的指针
if (*curErr) {
errorHandler();
*curErr = 0; // 正确: 把curErr所指的对象值重置
}
顶层const 与 底层const
指针本身是不是常量与指针所指量是不是一个常量是两个问题。
用顶层来表示指针本身是一个常量,用底层来表示指针所指对象是一个常量。
一般,用顶层const来表示常量,而底层const则与指针和引用等复合类型的基本类型相关。其中,以指针最为特殊,指针既可以是顶层const也可以是底层const。
int i = 0;
int *const p1 = &i; //不能改变p1的值, 这是一个顶层const
const int ci = 42; //不能改变ci的值, 这是一个顶层const
const int *p2 = &ci; //允许改变p2的值, 这是一个底层const
const int *const p3 = p2; //靠右的const是顶层const, 靠左的是底层const
const int &r = ci; //用于声明引用的const都是底层const
要注意,当进行对象拷贝的时候,顶层const不会发生改变而底层const会受到影响。
i = ci; //正确:拷贝ci的值, ci是一个顶层const, 对此操作无影响
p2 = p3; //正确:p2和p3指向的对象类型相同, p3顶层const部分不影响
底层const在拷贝时会有限制: 当执行对象的拷贝操作时,拷入和拷出的对象必须具有相同的底层const资格,或者两个对象的数据类型必须能够转换。 一般,非常量可以转换成常量,反之不行:
int *p = p3; //错误:p3包含底层const,而p没有
p2 = p3; //正确:p2和p3都具备const
p2 = &i; //正确: int* 能转换成const int*
int &r = ci; //错误: 普通的int&不能绑定到int常量上
const int &r2 = i; //正确:const int& 可以绑定到普通int上