【深入浅出指针】指针类型篇

80 阅读3分钟

持续创作,加速成长!这是我参与「掘金日新计划 · 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

image.png

指针数组

指针数组的概念:用于存放数组地址指针

创建形式: 类型 (*name)[size]

这里一不小心特别容易写成指针数组,这就是为什么我要在这一篇再一次提起指针数组的原因,它们二者的区别请看图:

image.png

数组名

实战先再解决一个问题,概念说数组指针是存放数组地址的,那如何取得数组地址呢?是我们之前文章提及的简简单单用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 )