【C语言】深度剖析指针和数组的关系


需要云服务器等云产品来学习Linux的同学可以移步/-->腾讯云<--/-->阿里云<--/-->华为云<--/官网,轻量型云服务器低至112元/年,新用户首次下单享超低折扣。 


 一、数组名和&数组名

二、字符指针

1、指向字符数组首元素的字符指针 char* p=arr

2、指向常量字符串的字符指针 const char* p="abc"

三、指针数组 int* p[3]={arr1,arr2,arr3}

四、数组指针 int(*p)[10]=&arr

五、数组传参

1、一维数组传参

2、二维数组传参

六、指针传参

1、一级指针传参

2、二级指针传参

七、函数指针

1、两个代码的理解

2、函数指针重命名

3、函数指针的使用——回调函数

八、函数指针数组——用途:转移表

九、函数指针数组指针


一、数组名和&数组名

区别:数组名表示首元素的地址,地址加1表示跳过一个数据类型的大小。

&数组名表示取出的是整个数组的地址,地址加1表示跳过一个数组的长度。

举例:arr[]需要使用整型(等)指针来指向,而&arr[]需要使用数组指针来指向。

数组名通常表示首元素的地址,但有两个例外:

1、sizeof(数组名),这里的数组名表示整个数组。

2、&数组名,表示取出整个数组。

二、字符指针

1、指向字符数组首元素的字符指针 char* p=arr

#include <stdio.h>
int main()
{
	char arr[5] = "abc";
	char* p = arr;
	printf("%s", p);//打印abc
	return 0;
}

数组名是首元素的地址,这里指针p指向的是arr数组的首元素'a'的地址,可以通过指针p,以%s的形式打印字符串。

2、指向常量字符串的字符指针 const char* p="abc"

#include <stdio.h>
int main()
{
	const char* p = "abc";//字符串abc为常量字符串
	printf("%s", p);
	return 0;
}

因为"abc"是常量字符串,不能被修改,所以指针p需要使用const修饰。若没有const,.c文件运行时程序会直接挂掉,.cpp文件更加严格,会直接报错。

三、指针数组 int* p[3]={arr1,arr2,arr3}

指针数组本质上是数组,是存放指针的数组。

#include <stdio.h>
int main()
{
	int arr1[] = { 1,2,3,4,5 };
	int arr2[] = { 2,3,4,5,6 };
	int arr3[] = { 3,7,5,8,99 };
	int* p[3] = { arr1,arr2,arr3 };
	for (int i = 0;i < 3;i++)
	{
		for (int j = 0;j < 5;j++)
		{
			//printf("%d ", p[i][j]);
            //printf("%d ", *(*(p+i) + j));//p[i]可以写成*(p+i)
			printf("%d ",*(p[i]+j));//解释如下图
		}
		printf("\n");
	}
	return 0;
}

三种指针数组的打印方式都是可以的,这里的代码类似二维数组,该数组首元素是第一行。详细解析如下:

四、数组指针 int(*p)[10]=&arr

数组指针是指向数组的指针。

int(*p)[10]=&arr,数组指针p的类型是int(*)[10](去掉p,即为类型)。这里指针+1跳过一个数组大小。

#include <stdio.h>
print(int(*p)[5], int r, int c)//接收的是指向一维数组的数组指针
{
	for (int i = 0; i < r;i++)
	{
		for (int j = 0; j < c; j++)
		{
			//printf("%d ", p[i][j]);
			//printf("%d ", *(p[i]+j));
			printf("%d ", *(*(p+i)+j));
		}
		printf("\n");
	}
}
int main()
{
	int arr[3][5] = { 1,5,4,6,7,9,8,4,2,5,4,5,4,5,6 };
	print(arr, 3, 5);//传过去的是二维数组
	return 0;
}

打印方式的理解与上一个例子相同,注意二维数组的首元素是第一行。

再来看一个例子:int(*p[10])[5],这是一个存放数组指针的数组,数组元素的类型是int(*)[5]。

五、数组传参

1、一维数组传参

void test(int arr[])//一维数组传参,数组接收
{}
void test(int arr[10])//一维数组传参,数组接收,10可有可无
{}
void test(int* arr)//一维数组传参,指针接收
{}
int main()
{
	int arr[10] = { 0 };
	test(arr);
}
void test2(int* arr[20])//指针数组传参,指针数组接收,20可有可无
{}
void test2(int** arr)//传过来的是一级指针的指针,当然可以用二级指针接收
{}
int main()
{
	int* arr2[20] = { 0 };
	test2(arr2);
}

2、二维数组传参

void test(int arr[3][5])//二维数组传参,二维数组接收
{}
void test(int arr[][])//错误,列不能省略
{}
void test(int arr[][5])//二维数组传参,二维数组接收
{}
void test(int *arr)//错误,传过来的是第一行的地址,相当于一维数组的地址,不能用一级指针接收
{}
void test(int* arr[5])//错误,不能用指针数组接收
{}
void test(int (*arr)[5])//可以用数组指针接收
{}
void test(int **arr)//错误,传过来的是一维数组的指针,不能用二级指针接收,二级指针是指向一级指针的指针
{}
int main()
{
 int arr[3][5] = {0};
 test(arr);//传过来的相当于第一行的地址
}

六、指针传参

1、一级指针传参

void text(int* p)
{}
int main()
{
	int a = 0;
	int* pa = &a;
	int arr[10] = { 0 };
	text(&a);//可以传整型变量的地址
	text(pa);//可以传一级指针的地址
	text(arr);//可以传整型数组的地址
	return 0;
}

只要传过来的本质是一级指针,就可以用一级指针接收。

2、二级指针传参

void text(int** p)
{}
int main()
{
	int* p1 = NULL;
	int** p2 = NULL;
	int* arr[10];
	text(&p1);//可以传一级指针的地址
	text(p2);//可以传二级指针
	text(arr);//可以传指针数组
	return 0;
}

七、函数指针

对于函数名与&函数名,它们的地址是一样的,且本质上没有任何区别。

使用函数值指针进行函数的调用,可以把p当成函数名调用,解引用无意义。

1、两个代码的理解

( *( void (*)() )0 )();

以上代码是一次函数调用,先将0强制类型转换为函数指针类型,该函数指针无参,返回值为void。由于函数指针解引用无意义,所以这里的这颗*可有可无。整个代码的意思是对0地址处的函数进行一次函数调用。

void ( *signal( int , void(*)(int) ) )(int);

以上代码是一次函数声明。

signal是函数名,它的参数是int和函数指针void(*)(int),返回类型是函数指针void(*)(int)。

2、函数指针重命名

typedef void(*pf_t)(int);
pf_t signal(int,pf_t);

把void(*)(int)类型重命名为pf_t。

上方函数可以用pf_t来代替void(*)(int)。

3、函数指针的使用——回调函数

int Add(int a, int b)
{}
int Sub(int a,int b)
{}
int Mul(int a, int b)
{}
int Div(int a, int b)
{}
int calc(int(*p)(int,int))
{
	return p(1, 2);
}
int main()
{
	calc(Add);
	return 0;
}

当有多个参数和返回类型均相同的函数时,可以使用函数指针对这些函数进行按需调用,减少代码冗余。

八、函数指针数组——用途:转移表

函数指针数组是存放函数指针的数组。

int Add(int a, int b)
{}
int Sub(int a,int b)
{}
int Mul(int a, int b)
{}
int Div(int a, int b)
{}
int main()
{
	int input,x,y;
    scanf("%d%d%d",&input,&x,&y);
    //int(*p)(int, int) = Add;
	int(*arr[4])(int,int) = {Add,Sub,Mul,Div};//数组arr的类型直接将函数指针的p替换即可
    int ret=arr[input](x,y);//使用函数指针数组对函数进行调用——转移表
	for (int i = 0; i < 4; i++)//函数指针数组的遍历调用
	{
		int ret = arr[i](8, 4);
    }
	return 0;
}

可以使用函数指针数组对相同形参、返回类型的函数进行调用,在实际使用中实现函数的跳转功能,所以被称为转移表,有较大的使用价值。

九、函数指针数组指针

int main()
{
    int(*p)(int, int) = Add;//p是函数指针
	int(*arr[4])(int,int) = {Add,Sub,Mul,Div};//arr是函数指针数组
    int(*(*parr))(int,int)=&arr//parr是函数指针数组指针
	return 0;
}

写法可以按照上一级指针/数组类型模仿写。

  • 96
    点赞
  • 100
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 82
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 82
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

蒋灵瑜

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值