持续创作,加速成长!这是我参与「掘金日新计划 · 10 月更文挑战」的第9天,点击查看活动详情
字符指针
从名字可以直接看出来,字符指针是用来存放字符或者字符串的指针
在指针类型中,我们提到过char* 类型,它就是字符指针:存放单个字符,或者存放一整个字符串的指针类型。
//存放字符的字符指针
char ch = 'w';
char* pc = &ch;
char *pch = "abcdef";
指针pc中存储了变量ch中‘w’字符的地址
对于第一次见到存储字符串的指针,我们首先要进行分析设问: ①“abcdef”字符串是存储到了指针变量pch中么?(假设是在32位环境下) 分析:在32位环境下,char*类型指针大小才4字节,而字符串“abcdef”有六个字节,明显存储不下,所以这个问题解决了。
②那存储的会是字符串的首元素‘a’的地址么? 通过代码来验证:
int main()
{
char *pch = "abcdef";
printf("%c\n",*pch);
return 0;
}
验证*pch中存储的地址是否是a
指针数组
指针数组的概念:用于存放数组地址的指针
创建形式: 类型 (*name)[size]
这里一不小心特别容易写成指针数组,这就是为什么我要在这一篇再一次提起指针数组的原因,它们二者的区别请看图:
数组名
实战先再解决一个问题,概念说数组指针是存放数组地址的,那如何取得数组地址呢?是我们之前文章提及的简简单单用arr代表整个数组的地址么?还是用* &arr?
整一个数组地址
①sizeof(arr)其中的arr代表整一个数组
②&arr,可以取出整一个数组的地址
首元素地址
平常中直接写出arr其代表数组中首元素地址
应用
//写一个打印函数,利用数组指针打印数组中的内容
int main()
{
int arr[10]={1,2,3,4,5,6,7,8,9,0};
int sz = sizeof(arr)/sizeof(arr[0]);//sizeof(arr)//计算整个数组大小
my_printf(&arr,sz);//&arr,整个数组的地址,为使用数组指针做准备
return 0;
}
//函数主题部分----本来这部分最好是在main函数前写,为了阅读体验而做出让步
void my_printf(int(*p)[10],int sz)
{
int i = 0;
for(i=0;i<sz;i++)
{
printf("%d ",*(*p+i));
}
}
上述代码详解:
p:指针中存储的是整个数组的地址
*p:对p进行解引用,得到的是arr,而arr又代表数组的首元素地址
*p+i:代表数组下标为i的元素地址
*(*p+i):代表下标为i的元素
但是!这样子运用起来其实特别别扭,我本来直接传arr首元素的地址还更加方便的打印,为何要这么绕来绕去,所以,数组指针一般不应用在一维数组中,反言之,数组指针一般应用在二维数组中。
//写一个打印函数,利用数组指针打印二维数组中的内容
int main()
{
int arr[3][5]={{1,2,3,,4,5},{2,3,4,5,6},{3,4,5,6,7}};
my_printf(arr,3,5);//arr,二维元素里面第一行元素的地址
return 0;
}
//函数主题部分----本来这部分最好是在main函数前写,为了阅读体验而做出让步
void my_printf(int(*p)[5],int c,intr)//C:行数,r:列数
{
int i = 0;
for(i=0;i<3;i++)
{
int j = 0;
for(j=0;j<5;j++)
{
printf("%d ",p[i][j]);//或者是*(*(p+i)+j)
}
printf("\n");
}
}
经过上面的例子我们可以对解引用做个初步的总结:对解引用的初步总结
arr[i] == * (arr+i)
arr[i][j] == *(arr+i)[j] == * ( * (arr+i)+j )