地址和指针
地址与指针变量是一体两面的。类似int型变量和int型数据的关系。在一般的地址传递中,类比数据传递,也是借助指针变量作为中介传递。没有指针变量,无法分配内存空间给地址数据,连存储都做不到,谈何传递。且利用指针变量作为中介也有助于理清思路和便于调试。
*(0x7338754) //*应接指针变量,这是一个错误。
以上的错误是需要避免的。
小仓老师说:
指针变量就是地址变量。
是很有道理的,且较精辟地指出的指针的本质。
数组和指针
数组名是指针,指向该数组第一个元素的首地址。但数组名属于const int *型指针,可改变它指向的内存空间的内容,却无法改变它的指向。
数组名作为指针使用时,永远指向其数组第一个元素的首地址。
int array[3]={1,2,5};
printf("%d %d %d\n",*array,*(&array+1),*(&array[0]+2));
1、3显示数字,2显示地址,讨论(&array+1)。
array显示地址;
&array显示地址;
(&array)显示地址;
(**&array)显示数值;
array作为数组名,值为地址,本身便含一层&。
数组名、&数组名、&数组名[0]是可以作为指向数组第一个元素首地址的指针使用的。
二维数组与指针
二维数组可分解为多个一维数组处理。
int array[3][4]={1,2,3,4,5,6,7,8,9,13,34,45};
array是整个二维数组的数组名,它的指向即可以说是指向整个二维数组的首地址,也可以说是指向第一行一维数组的首地址。
array[0]、array[1]、array[2]分别是二维数组下第一行、第二行、第三行的一维数组的数组名,作为数组名,它可以作为指向该数组的首地址的指针使用。
array作为整个二维数组的数组名,值为地址,本身含2层&。
故 array 显示地址;
array显示地址;
array显示值。
基于一维数组的认识,很容易得出array+i,array[i],(array+i),&a[i][0]是等同的。
可以分2个部分去理解,前3个是基于数组名的角度去认识,第4个是从首元素的首地址去认识指针。
再者array+i是从整个二维数组的角度(它是从整个二维数组的首地址出发,指针加法使用首地址,故这样说)使用array,array含2个&,array+i和(array+i)等同;而array[i]是将整个二维数组分解为一维数组处理了,这时array[i]就只含一个&了,array[i]与(array[i])不等同,一个是地址,一个是值。
((array+i)+j)是从整个二维数组出发,含2个&,显示为二维数组第i行第j列元素的首地址,(*(array+i)+j)是该元素的值。
指针数组与二维数组指针变量
指针数组:一组有序的指针的集合。
int *pa[3];
二维数组指针变量:单个变量,其一般形式中“(指针变量名)”两边的括号不可少。*
int (*pa)[3];
一般而言,指针将地址和值分得很清楚,在输入输出时也这样,地址归地址,值归值。但在字符输入输出时,%s%c下,scanf和printf会智能输入输出,ch为指向首地址的指针变量,会模糊,直接输入输出到首地址之后的字符数组中去。