前言:
大多数学习过C/C++或者正在学习的同学在对指针概念把握时,总是感觉不太明了,小弟我也不例外啊,于是翻开资料复习整理一下,并把自己的学习心得拿出来供大家分享,讨论。
基本概念掠过,主要来探讨一下容易混淆的几个概念。
1.指向常量的指针与指针常量
a.指向常量的指针
如const char *
pStr是指向常量的指针变量,既此定义的变量是一个指向字符数组的指针变量,我们可以对其进行再次赋值,但该指针指向的地址块的值不能改变。请看 下面的例子:
char ch[5]="lisi“; const char * pStr=ch; //pStr 是指向字符数组的指针变量 pStr="wangwu"; //这是没有任何问题的 *pStr="w"; //此处编译时会报错,因为我们通过指向*符找到pStr所指向的内存地址块是常量所占有,所以不能再次赋值。
指向常量的指针主要用于作为函数的参数,如定义一个画线函数:
void DrawLine(const CPoint *pt1,const CPoint * pt2){ // ..... }
通过参数pt1, pt2获得两个点的指针,在函数体中我们可以获取pt1->x,pt1->y,但是我们无法进行 pt1->x=3等赋值操作,从而保证了数据的一致性与安全性。
b.指针常量
下面我们对比性地来看指针常量 如:
char * const pStr;
此处定义的是一个指针常量,有的教材上叫它常量指针,个人觉得还是叫它指针常量比较容易理解记忆既然是常量,在定义时初始化之后,就不能对其再赋值,但其指向的 地址块的值可以改变。
char ch[5]="lisi"; char * const pStr=ch; pStr="zhangsan"; //错误 *pStr=‘w‘; //正确
由于const char * pStr与 char * const pStr外观上太相似,所以和容易弄混,我们可以通过下面的方法来记忆,但仅供参考,方便记忆而已,没有理论根据。
首先看char * const pStr中的 const修饰的是变量名pStr,所以pStr是指针常量而const char *
pStr中的const后面是(char * pStr), pStr已经被指向符 *
修饰 过,所以可以理解为const修饰的pStr所指向的地址块,该地址块为常量,所以const char *
pStr是指向常量的指针变量
如果你觉得上面讲还行,那就继续看下面的吧.
2.指针数组和指向数组的指针
a.数组
先我们先来探讨一下关于数组的问题,我们先定义一个数组
char ch[4]="abc";
我们可以对其进行
*(ch+0)=‘e‘;//等价于ch[0]=‘e‘;的合法操作
但是,接下来
char
ch2[4]="123";
ch=ch2; //编译时会报错
这是为什么呢?因为,数组名ch其实就是一个指向字符的指针常量,下面我们我们来做一个实验:
#include <stdio.h> #include <malloc.h> static void main() { char ch[4]="abc"; char * const pCh=(char *)malloc(4*sizeof(char)); //申请地址空间 并将说空间首地址赋给pCh *(pCh+0)=‘a‘; *(pCh+1)=‘b‘; *(pCh+2)=‘c‘; *(pCh+3)=‘\0‘; printf("%s %s",ch,pCh); }
如果你一眼看到char * const pCh就能说出pCh是一个指针常量的话,说明天才的你已经理解前面所讲的了,我感到万分荣幸。这个程序的结果是 :abc abc
例子中 char ch[4] 与 char * const pCh=(char *)malloc(5*sizeof(char)); 的效果是一样的,c内部处理上,个人认为应该也是一样的,所以说数组名本质就是就是一 个指针常量。
b.指针数组
很好理解 就是数组里存放的是指针 如:
char *
week[7]={"Monday","Tuesday","Wednesday","Thursday","Friday","Saturday","Sunday"}
该数组中存放的是指向字符串的指针
c.指向数组的指针
如:
int (*p)[3]; //
注意括号可不能丢了
那么p就是一个指向大小为3数组的指针变量,我们知道数组名就是指针,指向数组的指针就是一个二级指针。
int
a[][3]={1,2,3,4,5,6};
我们知道a是一个二维数组,可以进行以下操作
printf("%d
",*(*(a+0)+1));
printf("%d ",*(a[0]+2));
printf("%d
",a[1][2]);
结果分别是2 3
6
这是因为二维数组名是二级指针常量,接下来我们的最后一个小实验
#include <stdio.h> static void main() { int a[][3]={1,2,3,4,5,6}; int (*p)[3]=a; printf("%d ",*(*(p+0)+1)); printf("%d ",*(p[0]+2)); printf("%d ",p[1][2]); }
将会的到同样的结果2 3 6 。