天天看点

C语言指针(一)

int p; //这是一个普通的整型变量

int p; //首先从P处开始,先与结合,所以说明P是一个指针,然后再与int结合, 说明指针所指向的内容的类型为int型.所以P是一个返回整型数据的指针

int p[3]; // 首先从P处开始,先与[]结合,说明P是一个数组,然后与int结合,说明数组里的元素是整型的,所以P是一个由整型数据组成的数组

int p[3]; //首先从P处开始,先与[]结合,因为其优先级比高,所以P是一个数组, 然后再与*结合,说明数组里的元素是指针类型, 然后再与int结合, 说明指针所指向的内容的类型是整型的,所以P是一个由返回整型数据的指针所组成的数组

int (p)[3]; //首先从P处开始,先与结合,说明P是一个指针,然后再与[]结合 (与”()”这步可以忽略,只是为了改变优先级), 说明指针所指向的 内容是一个数组,然后再与int结合, 说明数组里的元素是整型的. 所以P是一个指向由整型数据组成的数组的指针

int *p; //首先从P开始,先与结合,说是P是一个指针,然后再与*结合, 说明指针所指向的元素是指针,然后再与int结合, 说明该指针所指向的元素 是整型数据.由于二级指针以及更高级的指针极少用在复杂类型中, 所 以后面更复杂的类型我们就不考虑多级指针了, 最多只考虑一级指针.

int p(int); //从P处起,先与()结合,说明P是一个函数,然后进入()里分析,说明该函数有一个整型变量的参数,然后再与外面的int结合, 说明函数的返回值是一个整型数据

Int (*p)(int); //从P处开始,先与指针结合,说明P是一个指针,然后与()结合,

说明指针指向的是一个函数,然后再与()里的int结合,说明函数有一个int型的参数,再与最外层的int结合,说明函数的 返回类型是整型,所以P是一个指向有一个整型参数且返回 类型为整型的函数的指针

int (*p(int))[3]; //可以先跳过,不看这个类型,过于复杂 从P开始,先与()结合,说明P是一个函数,然后进入()里面,与int结合,说明函数有一个整型变量参数,然后再与外面的结合,说明函数返回的是一个指针, 然后到最外面一层, 先与[]结合,说明返回的指针指向的是一个数组,然后再与*结合,说明数组里的元素是指针,然后再与int结合,说明指针指向的内容是整型数据.所以P是一个参数为一个整数据且返回 一个指向由整型指针变量组成的数组的指针变量的函数.

说到这里也就差不多了,我们的任务也就这么多,理解了这几个类型,其它的类型对我们来说也是小菜了,不过我们一般不会用太复杂的类型,那样会大大减小程序的可读性,请慎用,这上面的几种类型已经足够我们用了. 1、细说指针

指针是一个特殊的变量,它里面存储的数值被解释成为内存里的一个地址。 要搞清一个指针需要搞清指针的四方面的内容:指针的类型、指针所指向的 类型、指针的值或者叫指针所指向的内存区、指针本身所占据的内存区。让我们分别说明。

先声明几个指针放着做例子: 例一:

(1)int*ptr; (2)char*ptr; (3)int**ptr;

(4)int(ptr)[3]; (5)int(*ptr)[4];

1.1指针的类型

(1)int*ptr;//指针的类型是int*

(2)char*ptr;//指针的类型是char*

(3)int**ptr;//指针的类型是int**

(4)int(ptr)[3];//指针的类型是int()[3]

(5)int*(ptr)[4];//指针的类型是int(*)[4]

1.2指针所指向的类型

当你通过指针来访问指针所指向的内存区时,指针所指向的类型决定了编译器将把那片内存区里的内容当做什么来看待。

从语法上看,你只须把指针声明语句中的指针名字和名字左边的指针声明符*去掉,剩下的就是指针所指向的类型。例如:

(1)int*ptr; //指针所指向的类型是int ’

(3)int**ptr; //指针所指向的的类型是int*

(4)int(*ptr)[3]; //指针所指向的的类型是int()[3]

(5)int*(ptr)[4]; //指针所指向的的类型是int()[4]

在指针的算术运算中,指针所指向的类型有很大的作用。 指针的类型(即指针本身的类型)和指针所指向的类型是两个概念。当你对C越来越熟悉时,你会发现,把与指针搅和在一起的”类型”这个概念分成”指针的类型”和”指针所指向的类型”两个概念,是精通指针的关键点之一。我看了不少书,发现有些写得差的书中,就把指针的这两个概念搅在一起了,所以看起书来前后矛盾,越看越糊涂

1.3指针的值–或者叫指针所指向的内存区或地址

指针的值是指针本身存储的数值,这个值将被编译器当作一个地址,而不是一个一般的数值。在32位程序里,所有类型的指针的值都是一个32位整数,因为32位程序里内存地址全都是32位长。 指针所指向的内存区就是从指针的值所代表的那个内存地址开始,长度为sizeof(指针所指向的类型)的一片内存区。以后,我们说一个指针的值是XX,就相当于说该指针指向了以XX为首地址的一片内存区域;我们说一个指针指向了某块内存区域,就相当于说该指针的值是这块内存区域的首地址。

指针所指向的内存区和指针所指向的类型是两个完全不同的概念。在例一中,指针所指向的类型已经有了,但由于指针还未初始化,所以它所指向的内存区是不存在的,或者说是无意义的。

1.4指针本身所占据的内存区

指针本身占了多大的内存?你只要用函数sizeof(指针的类型)测一下就知道了。在32位平台里,指针本身占据了4个字节的长度。

指针本身占据的内存这个概念在判断一个指针表达式是否是左值时很有用。

左值 简单点说就是可以放在赋值运算符左边的表达式.下面让我们来看看他的定义: 如果一个表达式可以引用到某一个对象,并且这个对象是一块内存空间且可以被检查和存储,那么这个表达式就可以做为一个左值.当然,有左值当然就会有右值这个概念: 右值指的是引用了一个存储在某个内存地址里的数据。一个变量可以同时是左值,同时也是右值,两者不是对立的。

继续阅读