> 文章列表 > 【C语言】数组指针-c语言的任督二脉

【C语言】数组指针-c语言的任督二脉

【C语言】数组指针-c语言的任督二脉

视频链接:bilibili

关于指针需要注意的地方

只有以下两种情况数组名表示的是整个数组
1.sizeof(数组名)
2.&数组名
除此之外数组名表示的都是首元素地址

一、字符指针

是一个指向字符的指针

int main()
{char ch = 'w';char* p = &ch;//char* ch2 = "abcdef";//ch2里放的是字符串a的地址const char* ch2 = "abcdef";//常量字符串不允许被修改,int前面加const,修饰的是*ch2//char* const ch2 = "abcdef";//const修饰的是ch2这个地址return 0;
}

二、指针数组

是一个数组,是用来存放指针的数组

int main()
{int arr1[] = { 1,2,3,4,5,6 };int arr2[] = {2,3,4,5,6,7};int arr3[] = {3,4,5,6,7,8};int* parr[] = { arr1,arr2,arr3 };int i = 0;for (i = 0; i < 3; i++){int j = 0;for (j = 0; j < 5; j++){printf("%d ", *(parr[i] + j));}printf("\\n");}return 0;
}

三、数组指针

数组指针是一个指针,用来存放数组的指针

int *p1[10];//p1指的是一个存放指针的数组,这个数组中存储的是10个int类型的指针
int (*p2)[10];//p2指的是一个指针,指针指向数组,数组中有10个元素,括号优先级高于[]
int main()
{char* arr[5];char* (* pa)[5] = &arr;return 0;
}

在这里插入图片描述

void print1(int (*p)[5],int x, int y )
{int i = 0;for(i=0;i<3;i++){int j = 0;for(j=0;j<5;j++){//体会下面解引用的不同方法printf("%d ", *(*(p+i)+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,2,3,4,5},{2,3,4,5,6},{3,4,5,6,7}};print1(arr, 3, 5);//arr--数组名--数组名就是首元素地址return 0;
}

思考下面代码表示是什么意思

int arr[5];
int *parr1[10];
int (*parr2)[10];
int (*parr3[10])[5];

四、数组传参

一维数组传参

void test(int arr)//不行
void test(int arr[5])//可以
void test(int arr[])//可以
void test(int* arr[])//可以
void test1(int* arr[])//可以
void test1(int* arr[5])//可以,arr2是一个指针数组,既然是数组就可以用指针接收
void test1(int** arr)//可以,arr2内存储是指针,那么可以用二级指针接收int main()
{int arr[5] = { 0 };int* arr2[5] = { 0 };//指针数组,数组的每个元素是指针,arr2表示的是首元素地址,//可以用首元素地址的地址来接收test(arr);test1(arr2);return 0;
}

二维数组传参
二维数组传参时,行可以省略,列不可省略

void test(int arr[3][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);return 0;
}

五、函数指针

指向函数的指针, 需要明确函数也有地址
&函数名和函数名都是函数的地址。而函数指针在调用的时候可以使用解引用也可以不解引用

int Add(int x, int y)
{int z = x + y;return z;
}
int main()
{int a = 10;int b = 20;//printf("%d\\n", Add(a, b));//printf("%p\\n", &Add);int (*pa)(int, int) = Add;//将函数Add的地址存到pa这个函数指针中。printf("%d\\n", (*pa)(2,3));return 0;
}
**思考下面代码的含义**
(*(void (*)())0)();

六、函数指针数组

函数指针数组,是一个数组,存放的是函数的地址

int Add(int x, int y)
{return x + y;
}
int Sub(int x, int y)
{return x - y;
}
int main()
{//int (*pa)(int, int) = Add;int (*pa[2])(int, int) = {Add, Sub};int i = 0;for (i = 0; i < 2; i++){printf("%d\\n", pa[i](2, 3));//5  -1  函数指针在调用的时候可以使用解引用也可以不解引用}return 0;
}

函数指针数组—也可以称为转移表

//实现一个计算器的功能
int Add(int x, int y)
{return x + y;
}
int Sub(int x, int y)
{return x - y;
}int Mul(int x, int y)
{return x * y;
}int Div(int x, int y)
{return x / y;
}
void menu()
{printf("----------------------------------\\n");printf("---------1.Add   2.Sub -----------\\n");printf("---------3.Mul   4.Div -----------\\n");printf("--------     0.exit   ------------\\n");printf("----------------------------------\\n");
}
int main()
{int input = 0;int a = 0;int b = 0;do{menu();printf("请选择:>");scanf_s("%d", &input);printf("请输入两个操作数:>");scanf_s("%d%d", &a, &b);//paArr是一个函数指针数组--转移表int (*paArr[])(int, int) = {0, Add, Sub, Mul, Div};if (input>=1 && input<=5){int ret = paArr[input](a,b);printf("%d\\n", ret);}else if (input == 0){printf("退出");break;}else{printf("输出错误");}} while (input);return 0;
}

七、指向[函数指针数组]的指针

int main()
{int arr[5] = { 0 };//pf是一个函数指针int (*pf)(int,int);int(*pf)[5] = &arr;//p是一个数组指针//pfArr是一个数组//是一个函数指针数组,包含5个元素//每个元素的类型是一个函数指针//函数指针的类型是int (*)(int,int)int (*pfArr[5])(int,int);//ppfArr是一个指针//是一个数组指针,指向的数组有5个元素//每个元素的类型是一个函数指针//函数指针的类型是int (*)(int,int)int (*(*ppfArr)[5])(int, int) = &pfArr;//ppfArr是一个指向[函数指针数组]的指针return 0;
}

八、回调函数

通过函数指针调用的函数,用于解决代码冗余的问题,如果把函数的地址作为参数传递给另一个函数,

#include <stdio.h>
//设计一个main函数,用于调用一个回调函数
//设计一个print函数,实现打印字符串的功能
//设计一个回调函数,用于接收函数地址,
//并且在回调函数内部调用print函数
void print(char *str)//print函数是打印字符串的
{printf("%s\\n", str);
}void test(void (*p)(char*))//test函数是一个回调函数
{printf("haha\\n");p("bit");//调用test接收到的函数指针,找到对应的函数进行调用
}int main()
{test(print);return 0;
}

设计前沿