对于非计算机专业的同学,c语言的指针往往就是老师的一句“指针不考“就带过了。c语言的指针号称是c语言的灵魂,是c语言中最精妙的部分。
指针本质上也是变量,也就是一段内存,只是他的特殊之处是他存储的数据是地址。地址又是什么概念?每一段内存都有其相应的地址,用街道来比喻的话,地址就是门牌号,数据就储存在家里。指针变量的作用就是储存门牌号。
定义一个指针变量要加一个*号。这里要提及两个概念,指针的类型和指针指向的类型。首先来说一下指针的类型。一般来说,指针类型要与指针指向的类型相匹配,不匹配的话往往会有警告。但是指针类型可以不匹配指针指向的类型。指针类型可以是用原本存在的类型如int *,char *,也可以自己定义,如结构体指针等。我认为定义指针类型目的就是确定指针的步长,即对指针变量进行p++操作时,p的数据加了几个字节。如int *p进行p++时,p的数据加了四个字节(对于32位系统),因为int型占的内存就是4个字节。所以同样的,char *p进行p++操作时,p的数据加了一个字节。同样的,定义一个占20个字节的结构体stu,那么stu *p的p++里面数据增加20个字节。所以就可以用char *p来访问int类型中每一个字节的数据。程序和运行结果如下:
所以可以通过这样的方法来访问int定义的i中每一个字节所存储的内容,也可以通过这样的方法确定大小端。
接下来说一下指针指向的类型。指针的作用保存一段内存的地址,也就是指向这段内存。所以指针指向的类型就是指针变量保存的地址上的那段内存的类型。
实际上指针变量p也同样占有一段内存,因此指针变量p也有地址,可以定义一个指向指针的指针。如int *p1=NULL;int **p2=&p1;其中p2就是指向指针的指针。
还有一个需要注意的是指针变量的长度只和系统有关,32位系统中无论定义什么样类型的指针,指针变量所占的内存都是32位,也就是四个字节;64位系统的指针都是八个字节。定义指针类型的目的就是为了确定指针的步长,目的是为了更方便地读取数据。如定义一个int *p1=&a;char *p1=&a;这里面p1和p2所存的数据是相同的,p1和p2的所占用的内存也是相同的,但是进行p1++和p2++后,他们就相差3个字节,这就是步长的意义,也就是定义不同类型的指针变量的意义。下面举个栗子来说明一下:
最后来说一下关于指针的几种常见形式和意义
int *p=NULL //定义一个指针变量p,类型为int *,该指针步长为4(因为对于32位系统来说,int型数据的占用内存是四个字节),指向的地址为空。一般定义一个指针如果不赋初值的话,为了安全要指针指向NULL。
p=&a //将变量a的地址赋值给p。变量的本质是一段内存,内存都有地址。a是一个变量,也就是一段内存,所以有地址。
a=*p //将p指向的内存的数据赋值给a。除了在定义时,*p的意义就是p指向的那段内存存的数据。
typedef struct stu{}*p_st;p_st p1; //定义一个结构体变量p1,p1的步长为结构体所占的字节。
来源:https://www.cnblogs.com/wangyouwei/p/5680820.html