什么是指针,我们在之前的《指针》章节已经接触过了,我们知道了指针的概念:
1.指针就是个变量,用来存放地址,地址唯一标识一块内存空间。
2指针的大小是固定的4/8个字节( 32位平台/64位平台)。
3.指针是有类型,指针的类型决定了指针的+-整数的步长,指针解引用操作的时候的权限。
4.指针的运算。可以比较大小,如果两个指针指向同一数组,也可以相减,得到数的绝对值是之间的元素个数。
1.字符指针
在指针类型中,我们知道有一种指针类型为字符指针 char* ;
一般使用情况:
int main() { char ch = 'w'; char* pc = &ch;//pc就是字符指针,指向字符的指针就是字符指针 *pc = 'w'; return 0; }
还有一种使用情况如下:存放常量字符串的首字符的地址
#include<stdio.h> int main() { //下面右边是一个表达式,表达式的值是首字符的地址 char* p = "abcdef";//是把首字符的地址放到p中 //*p = 'w';//错误,这里字符串是常量字符串,不能修改 //const char *p = "abcdef";//这样写会更加严谨, char arr[] = "abcdef"; //可以想象为字符串是arr,p是存放的arr的首元素地址 char* parr = arr; // p指向的是arr首元素地址,不同的是arr数组是可以修改 *parr='w';//可以修改arr字符串的值 printf("%s\n", arr); return 0; }
代码 char* p = "abcdef"; 特别容易让我们以为是把字符串 abcdef 放到字符指针 p 里了,但是/本质是把常量字符串 abcedf 首字符的地址放到了p中,也就是a的地址放到了指针变量p中。
看一下面这道题
#include <stdio.h> int main() { char str1[] = "hello xilanhua"; char str2[] = "hello xilanhua"; const char* str3 = "hello xilanhua"; const char* str4 = "hello xilanhua"; if (str1 == str2) { printf("str1 和 str2 相等\n"); } else { printf("str1 和 str2 不相等\n"); } if (str3 == str4) { printf("str3 和 str4 相等\n"); } else { printf("str3 和 str4 不相等\n"); } return 0; }
str1 和str2 是两个字符数组的首元素地址,在内存中开辟不同的空间,所以str1和str2不相等,
str3 , str4 存放常量字符串 "hello xilanhua" 的首字符地址,常量字符串不能被修改,所以在内存中没有必要存两份,只需要开辟一份空间,两个指针都指向的这份空间,所以两个地址相同的。
所以输出结果为
2.指针数组
可以通过类比:
整型数组,存放整形的数组
字符数组,存放字符的数组
指针数组,存放指针的数组。
int* arr1[5];//整形指针数组 char* arr2[5];//一级字符指针的数组 char** arr3[5];//二级字符指针的数组
根据上面学的字符指针,我们看一下下面代码:
#include <stdio.h> int main() { char* arr[] = { "abcdef","hehe","xilanhua" };//字符指针数组 // arr0 arr1 arr2 //这里存放的是常量字符串的首字符的地址 //*arr[1] = 'w';//错误,这里数组元素指向的也是常量字符串,不能修改 for (int i = 0; i < 3; i++) { printf("%s\n", arr[i]); } return 0; }
结果
看这段代码:
#include<stdio.h> int main() { int arr1[] = { 1,2,3,4,5 }; int arr2[] = { 2,3,4,5,6 }; int arr3[] = { 3,4,5,6,7 }; //数组名是首元素地址 //arr是存放整形指针的数组 指针数组 int* arr[] = { arr1,arr2,arr3 }; int i = 0; int j = 0; for (i = 0; i < 3; i++) { for (j = 0; j < 5; j++) { printf("%d ", arr[i][j]);//和二维数组很像 //printf("%d ", *(arr[i] + j)); //*(arr[i]+j)==arr[i][j] arr[i]==*(arr+i) } printf("\n"); } return 0; }
结果:
可以发现和二维数组很像,但是本质上是不同的,二维数组是连续存放的空间,但是这里不是,是通过指针联系起来的。
3.数组指针
也是通过类比:
整形指针,指向整形的指针变量
字符指针,指向字符的指针变量
数组指针,指向数组的指针变量
下面哪一个是数组指针?
int* p1[10];
int (*p2)[10];
解释:
int (*p)[10]
p先和*结合,说明p是一个指针变量,然后指着指向的是一个大小为10个整型的数组。所以p是一个指针,指向一个数组,叫数组指针。
这里要注意:[ ] 的优先级要高于*号的,所以必须加上()来保证p先和*结合。不加(),就是指针数组。
1. &数组名 和 数组名
看下面代码:
#include<stdio.h> int main() { int arr[10] = { 0 }; printf("%p\n", arr);//数组名是首元素地址 两个例外 &数组名和sizeof(数组名) printf("%p\n", &arr[0]);//首元素地址 printf("%p\n", &arr); //三个数值上相等 printf("%p\n", arr + 1);//int* + 1 printf("%p\n", &arr[0] + 1);//int* + 1 printf("%p\n", &arr + 1);//加了40 int(*)[10]+1 int(*p)[10] = &arr; //int(*)[10] 这个数组指针的类型 return 0; }
根据上面的代码我们发现,其实&ar r和 arr,虽然值上是相同的,但是意义是不相同的。实际上:&arr表示的是数组的地址,而不是数组首元素的地址。(细细体会一下)。数组的地址+1,跳过整个数组的大小,所以 &arr+1 相对于&arr 的差值是40.
总结:
数组名是首元素地址 有两个例外 &数组名 和 sizeof(数组名)
1.sizeof(arr) - sizeof内部单独放一个数组名的时候,数组名表示整个数组,计算得到的是数组的大小
2.&arr - 这里的数组名表示的是整个数组,取出的是整个数组的地址,从地址值的角度来讲和数组的首元素地址是一样的,但是意义不一样
2.访问数组元素的不同方式
#include<stdio.h> int main() { int arr[10] = { 1,2,3,4,5,6,7,8,9,10 }; int i = 0; int sz = sizeof(arr) / sizeof(arr[0]); //下标形式访问数组 for (i = 0; i < sz; i++) { printf("%d ", arr[i]); } //使用指针访问数组 int* p = arr; for (i = 0; i < sz; i++) { for (i = 0; i < sz; i++) { printf("%d ", *(p + i)); } } //数组指针访问数组 虽然对,但是不推荐 int(*pa)[10] = &arr; //pa == &arr //*pa == *&arr //*pa == arr 数组名 不是两种特殊情况 是首元素地址 for (i = 0; i < sz; i++) { printf("%d ", *((*pa)+i));//(*pa)[i]; } return 0; }
注意:
int(*pa)[10] = &arr;
//pa == &arr
//*pa == *&arr
//*pa == arr 数组名 不是两种特殊情况 是首元素地址所以:数组指针 解引用 是数组首元素地址
3. 数组指针的使用
二维数组的传参要使用到指针数组。
#include<stdio.h> //普通接收二维数组传参形式 void print1(int arr[3][5], int r, int c) { int i = 0; int j = 0; for (i = 0; i < r; i++) { for (j = 0; j < c; j++) { printf("%d ", arr[i][j]); } printf("\n"); } } //二维数组传参用到数组指针 void print(int (*arr)[5], int r, int c)//指针形式接受 数组指针 { int i = 0; int j = 0; for (i = 0; i < r; i++)//arr+i是每一行 *(arr+i)是每一行的数组名 arr[i] 首元素地址就是每一行的第一个元素的地址 { for (j = 0; j < c; j++) { //printf("%d ", *(*(arr + i) + j));//解引用数组指针就是数组首元素地址,二维数组是第一行地址 printf("%d ", arr[i][j]);//*(*(arr+i)+j)==*(arr+i)[j]==arr[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 }; print(arr, 3, 5);//二维数组的数组名,也表示首元素地址 //二维数组的首元素地址是第一行的地址,存放的是一维数组 //可以理解为二维数组是存放一维数组的数组 return 0; }
二维数组的数组名,也表示首元素地址。二维数组的首元素地址是第一行的地址,存放的是一维数组。可以理解为二维数组是存放一维数组的数组
4.数组传参和指针传参
在写代码的时候难免要把【数组】或者【指针】传递给函数,那函数的参数该如何设计呢?
一维数组传参:
形参可以是数组,也可以是指针,当形参是指针时,要注意类型。
#include<stdio.h> void test(int arr[])//不会真的创建一个数组,不写大小 {} void test(int arr[10])//和传入格式一样,不会真的创建 {} void test(int* arr)//数组名,首元素地址,指针接受 {} void test2(int* arr[20])//指针数组,和传入格式一样 {} void test2(int** arr )//数组首元素是int* 类型,*arr说明他是指针 {} int main() { int arr[10] = { 0 };//整形数组 int* arr2[20] = { 0 };//整形指针数组 test(arr); test2(arr2); return 0; }
二维数组传参:
参数可以是指针,也可以是数组,如果是数组,行可以省略,列不能省略,如果是指针,传参传过去的是第一行的指针,形参就应该是数组指针
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); return 0; }
一级指针传参:
#include<stdio.h> void print(int* p, int sz) { int i = 0; for (i = 0; i < sz; i++) { printf("%d\n", *(p + i));//通过指针访问 } } int main() { int arr[10] = { 1,2,3,4,5,6,7,8,9,10 }; int* p = arr;//一级指针 int sz = sizeof(arr) / sizeof(arr[0]);//元素个数 print(p, sz); return 0; }
思考:当一个函数参数是一个指针时,函数能接收什么参数
void print(int* p);
1.int a;
print(&a);//变量的地址
2.int* p=&a;
print(p);//指针变量
3.int arr[10];
print(arr);//数组名
二级指针传参:
#include<stdio.h> //用二级指针接收 void test(int** ptr) { printf("num = %d\n", **ptr); } int main() { int n = 10; int* p = &n; int** pp = &p; test(pp); test(&p); return 0; }
思考:当函数的参数为二级指针的时候,可以接收什么参数
1.一级指针的地址
2.二级指针变量,
3.指针数组
5.函数指针
这里还是类比:
整形指针,指向整形的指针 int*
字符指针,指向字符的指针 char*
数组指针,指向数组的指针,int arr[10]; int (*p)[10] = &arr;
函数指针,指向函数的指针, 即存放函数的地址
那么问题来了,函数有地址吗
#include<stdio.h> int Add(int x, int y) { return x + y; } //&函数名得到就是函数的地址 int main() { printf("%p\n", &Add);//可以运行 printf("%p\n", Add);//可以运行 return 0; } //函数名 和 &函数名都是函数的地址
结果:
可以得出结论,函数名 和 &函数名 都是函数的地址。
使用函数指针调用函数:
#include<stdio.h> int Add(int x, int y) { return x + y; } int main() { //()是pf先于*结合,说明pf是指针 int (*pf)(int, int) = Add;//函数的地址要存起来,就得放在 函数指针变量pf 中 //最前面的是函数返回类型,后面括号内是函数参数类型 //通过函数指针调用函数 int ret = (*pf)(3, 5); ret = Add(3, 5);//函数可以这样调用,Add是地址 ret = pf(3, 5);//所以这样写也对 ret = (*****pf)(3, 5);//编译器在处理时,会把*去掉,也没有问题 printf("%d\n", ret);//8 return 0; }
可以阅读两段有趣的代码
1.(*(void(*)( ))0)( );
#include<stdio.h> int main() { //1.将0强制类型转换为void(*)()类型的函数指针 //2.这就意味着0地址处放着一个函数,函数无参,返回类型是void //3.调用0地址处的这个函数 (*(void(*)())0)();//前面*可以不写 return 0; }
2.void(* signal(int, void(*)(int) ) )(int);
int main() { void(* signal(int, void(*)(int) ) )(int); //signal先于()结合 //void(*)(int) //signal(int, void(*)(int));//函数名称和函数参数类型 //上述代码是一个函数的声明, //函数的名字是signal //函数的参数第一个是int类型,第二个是void(*)(int)类型的函数指针 //该函数指针指向的函数参数是int,返回类型是void //signal函数的返回类型也是一个函数指针 //该函数指针指向的函数参数是int,返回类型是void return 0; } 可以通过类型重命名简化为: //typedef int(*)(int) pt_f;//写法不对 typedef void(*pf_t)(int);//将void(*)(int)类型的函数指针重命名叫pf_t pf_t(signal(int, pf_t));
6.函数指针数组
存放函数指针的数组
函数指针数组写法
函数指针 int (*p)(int,int); 函数指针数组只需在p后加一个[大小],让指p先于[]结合 int (*p[10])(int,int); //数组存放元素的类型就是 int (*) (int,int)
函数指针数组的用途:转移表
例子:(计算器)
#include<stdio.h> 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"); } int main() { //转移表 - 函数指针数组 函数指针类型要相同 int(*pfArr[5])(int, int) = { NULL,Add,Sub,Mul,Div };//NULL==0 // 下标 0 1 2 3 4 int input = 0; int x = 0; int y = 0; int ret = 0; do { menu(); printf("请输入:> "); scanf("%d", &input); if (input == 0) { printf("退出计算器\n"); break; } if (input <= 4 && input >= 1) { printf("请输入两个操作数:>"); scanf("%d %d", &x, &y); printf("%d\n",pfArr[input](x, y));//调用相应的计算函数 } else { printf("输入非法,请重新输入\n"); } } while (input); return 0; }
7.指向函数指针数组的指针
指向函数指针数组的指针是一个指针,指针指向一个数组,数组的元素都是函数指针。
示例:
#include<stdio.h> int Add(int x, int y) { return x + y; } int Sub(int x, int y) { return x - y; } int main() { //函数指针 int(*pf)(int, int) = Add; //函数指针数组 int(*pfArr[4])(int, int) = { Add,Sub }; //ppfArr是一个指向一个函数指针数组的指针变量 int(*(*ppfArr)[4])(int,int) = &pfArr; return 0; }
8.回调函数
回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。
这里演示一个用到回调函数的库函数,qsort 函数,包含头文件 stdlib.h 。
我们通过c++官方网站旧版上可以查到 qsort 函数的用法qsort用法
先看返回值类型和参数列表
它可以排序任何类型的数据,但是需要自己写一个函数用来可以确定两个元素的大小,也就是参数列表中的compar函数。
void qsort( void* base,//待排序数组的第一个元素 size_t num, //待排序的元素个数 size_t size, //每个元素的大小 int(*cmp)(const void*, const void*));//函数指针,指向一个函数,这个函数可以比较2个元素的大小
qsort 函数底层使用的是快速排序,也是一种排序算法,我们这里用冒泡排序实现一下。
因为这个函数可以排序任何类型,所以我们最开始用 void * 指针来接收传来的指针,然后再通过强制类型转换 (char*) 再加上 size 来确定操作空间的大小。
我们先来实现以下cmp 函数
这是cmp函数的要求,返回值有大于0,小于0,和等于0,三种情况。 大于0指向p1,p1大,小于0指向p2,p2大,相等返回值是0。
int cmp_int(const void* p1, const void* p2) { return *(int*)p1 - *(int*)p2; //这是我们自己写的函数,我们知道是int类型,所以强制类型转换为(int*) }
交换函数:
因为我们不知道要交换什么类型元素,所以是使用 (char*) 强制类型转换,每次交换一个字节,循环size次的方法
void Swap(char* p1, char* p2, size_t size) { for (size_t i = 0; i < size; i++) { char t = *(p1 + i); *(p1 + i) = *(p2 + i); *(p2 + i) = t; } }
排序内部:
void bubble_sort(void* base, size_t num, size_t size, int(*cmp)(const void*, const void*)) { size_t i = 0; size_t j = 0; for (i = 0; i < num - 1; i++)//一趟冒泡排序 { for (j = 0; j < num - 1 - i; j++) { //两个相邻元素比较 //arr[j]与arr[j+1] 调用我们自己写的比较函数,回调函数 if (cmp((char*)base + j * size, (char*)base + (j + 1) * size) > 0) { //交换 Swap((char*)base + j * size, (char*)base + (j + 1) * size,size); } } }
全部代码:
#include<stdio.h> #include<stdlib.h> int cmp_int(const void* p1, const void* p2) { return *(int*)p1 - *(int*)p2; //这是我们自己写的函数,我们知道是int类型,所以强制类型转换为(int*) } void Swap(char* p1, char* p2, size_t size) { for (size_t i = 0; i < size; i++) { char t = *(p1 + i); *(p1 + i) = *(p2 + i); *(p2 + i) = t; } } void bubble_sort(void* base, size_t num, size_t size, int(*cmp)(const void*, const void*)) { size_t i = 0; size_t j = 0; for (i = 0; i < num - 1; i++)//一趟冒泡排序 { for (j = 0; j < num - 1 - i; j++) { //两个相邻元素比较 //arr[j]与arr[j+1] if (cmp((char*)base + j * size, (char*)base + (j + 1) * size) > 0) { //交换 Swap((char*)base + j * size, (char*)base + (j + 1) * size, size); } } } } int main() { int arr[10] = { 1,2,8,9,10,5,3,6,4,7 }; qsort(arr, 10, sizeof(arr[0]), cmp_int); for(int i=0;i<10;i++) { printf("%d ", arr[i]); } printf("\n"); int arr1[10] = { 1,2,8,9,10,5,3,6,4,7 }; bubble_sort(arr1, 10, sizeof(arr[0]), cmp_int); for (int i = 0; i < 10; i++) { printf("%d ", arr[i]); } return 0; }
本篇结束