有一段时间没有总结了,一是最近比较忙(借口,时间就像海绵,挤挤还是会冒水的),而来确实还没有进入真正的简书习惯。不废话了,进入正题,总结下指针的学习,本科就接触过这东西,可惜那时根本没有认真,其实那时学校就是个一般二本,以为读了也是白读,其实不是的,其实在任何学校,只要你愿意,你可以学到基本的很多一些东西,现在来说,以前一知半解的东西可以好好学学了。
说到指针,其实很多人会想到指南针,那么指南针其实是用来指明方向的,当然指针也是用来指明方向和地址的,说白了,一个指针也就是内存里的地址,用来存储数据的,我们可以利用指针所指内存空间去随意调用数据,非常方便。
1>一维数组的定义
*:作为指针的标志,以前第一次学的时候基本是走过场,这次学习之前很多学生一直纠结指针不好理解,对此可能因人而异吧。首先如何去定义一个指针,对于定义一个指针,主要包括类型说明符+"*"+变量名。
例如:定义一个整型的指针变量如下:
int *p=&a;
p=&a或者p=&a[0];
在这里是把一个数a的地址给了p,这样就可以通过地址p来调用a这个元素。
又如定义了一个一维数组arr,则p=&arr[0],数组首个元素的地址就是该数组的地址。
int arr[]={1,2,3,4,5,6};
for(i=0;i<6;i++)
{
printf("arr[%d]=%d\n",i,*(arr+i));
printf("arr[%d]的地址=%p\n",i,arr+i);
}
当然获得数组的每个元素还可以通过指针变量p来寻址,如
printf("arr[%d]的地址=%p\n",i,p++);
这样就可以输出所有元素在内存的地址。
2>二维数组的定义:数据类型(int float char double),数组名[行的大小][列的大小]
当然对于二维数组的定义,我们可以看作是多个一维数组构成二维数组的定义方式有以下几种:
1>二维数组的初始化
int arr1[2][3]={{1,2,3},{4,5,6}};
2>如果没有完全数组元素,用0补全
int arr2[2][3]={{1,2},{4,5}};
3>int arr3[2][3]={1,2,3,4,5,6};
4>int arr4[2]['a']={1,2,3};
5>根据后面的列数推算行数大小
int arr5[][3]={1,2,3,4,5};
基于第五种方式,以下写法是错误的
int arr6[2][]={1,2,3,4};
6>int arr6[2][3]={{[1]=2},{[1]=5}};
2.1>二维数组定义时需要注意的点位:
1>行号和列号不能以变量存在,必须要以常量存在;
2>行号可以省,列号不能省;
3>二维数组的访问
数组元素的访问:数组名[行下标][列下标],这可能是最常见的访问方式,并且0<=行下标<行的大小,0<=列下标<列的大小,如对arr1进行访问,访问arr1任何一个元素,则只需要在主函数里输入以下代码:
for(i=0;i<2,i++)
{
for(j=0;j<3,j++)
{
printf("arr1[%d][%d]=%d\t",i,j,arr1[i][j]);
}
printf("\n");
}
其实二维数组在内存中是连续存储的,可以用以下代码证明:在上述代码中加这句
printf("arr1[%d][%d]=%p\t",i,j,&arr1[i][j]);
除了以上寻址方式,二维数组还有多种方式,对于访问第i行第j列的地址,
&arr1[i][j]=arr1[i]+j=*(arr1+i)+j
而对应的第i行第j列的值是arr1[i][j]=*(arr1[i]+j)=*(*(arr1+i)+j)
二维数组名保存的是第0行第0列的元素的地址
第i行第0列元素的地址:&arr1[i][0],arr1+i,arr1[i],*(arr1+i)
4>指向多维数组的指针变量
可定义为:int (*p)[4]
先计算(),表明它是一个指针变量,再算[],代表这个指针将来指向一个大小为4的一维数组。
int *p[4]
先算[],表明它是一个数组,再算*,表明将来这个数组里面存的是指针变量
int a[2][3]={1,2,3,4,5,6,7,8,9};
int *pa[3]={a[0],a[1],a[2]};
int *p=a[0]
这样pa是一个指针数组,a[0],a[1],a[2]均为指针变量。