知识点回顾:
1. const 关键字
在 C 语言中,有一种能力可以将变量变成具有常量一样的特性。这就是 —— const 关键字。
在它的修饰下,变量就会失去可修改的特性,也就是变成只读的属性。
const int price = 520; const char a = 'a'; const float pi = 3.14;
2. 指向常量的指针
万能的指针当然也可以指向被 const 修饰过的变量,这就意味着不能通过指针来修改它所引用的值。
这时候,如果尝试修改指针引用的值,那么我们将被告知程序无法通过编译;但如果只是修改指针的指向,那么编译器并不会阻止你这么做。
#include <stdio.h>
int main()
{
int num = 520;
const int cnum = 880;
const int *pc = &cnum;
printf("cnum: %d, &cnum: %p\n", cnum, &cnum);
printf("*pc: %d\n, pc: %p", *pc, pc);
*pc = 1024; // 尝试修改 *pc 的值,报错
printf("*pc: %d\n", *pc);
return 0;
}
总结:
· 指针可以修改为指向不同的常量
· 指针可以修改为指向不同的变量
· 可以通过解引用来读取指针指向的数据
· 不可以通过解引用修改指针指向的数据
指向非常量的常量指针
指向常量的指针,不能改变的是指针指向的值,但指针本身是可以被修改的。如果要让指针也不可变,那么可以使用常量指针。
同样是使用 const 关键字修饰即可,只是位置稍微发生了变化:
#include <stdio.h>
int main()
{
int num = 520;
const int cnum = 880;
int *const p = #
*p = 1024; // 这是可以的
printf("*p: %d\n", *p);
p = &cnum; // 这是禁止的
printf("*p: %d\n", *p);
return 0;
}
它有如下特性:
· 指针自身不可以被修改
· 指针指向的值可以被修改
指向常量的常量指针
就是在刚才的基础上进一步限制,让常量指针指向的值也是常量:
#include <stdio.h>
int main()
{
int num = 520;
const int cnum = 880;
const int *const p = &cnum;
*p = 1024;
printf("*p: %d\n", *p);
p = &cnum;
printf("*p: %d\n", *p);
return 0;
}
这样指针自身不能被改变,它所指向的数据也不能通过对指针进行解引用来修改。其实这种霸气十足的限制在平时很少派上用场啦。并且我们发现——如果初始化时,指向的对象不是 const 修饰的变量,那么我们仍然可以通过变量名直接修改它的值:
#include <stdio.h>
int main()
{
int num = 520;
const int cnum = 880;
const int *const p = #
printf("*p: %d\n", *p);
num = 1024;
printf("*p: %d\n", *p);
/*
*p = 1024;
printf("*p: %d\n", *p);
p = &cnum;
printf("*p: %d\n", *p);
*/
return 0;
}
总结:
· 指针自身不可以被修改
· 指针指向的值也不可以被修改
指向“指向常量的常量指针”的指针
标题看起来似乎挺恐怖的,但其实你只要仔细思考,就不难发现:关于指针的那点事儿,永远都是换汤不换药的把戏。
#include <stdio.h>
int main()
{
int num = 520;
const int cnum = 880;
const int *const p = &cnum;
const int *const *pp = &p;
printf("pp: %p, &p: %p\n", pp, &p);
printf("*pp: %p, p: %p, &cnum: %p\n", *pp, p, &cnum);
printf("**pp: %d, *p: %d, cnum: %d\n", **pp, *p, cnum);
return 0;
}




