正文开始前我们先来回忆一下C语言指针详解(上)的一些重点:

  1. 指针就是个变量,用来存放地址,地址唯一标识一块内存空间。
  2. 指针的大小是固定的4/8个字节(32位平台/64位平台)。
  3. 指针是有类型,指针的类型决定了指针的±整数的步长,指针解引用操作的时候的权限。
  4. 指针 - 指针计算的是指针间的元素个数。

下面开始我们的正文

文章目录:

  • 1.字符指针
  • 2.指针数组和数组指针
    • 2.1指针数组
    • 1.2数组指针
    • 1.3&数组名和数组名
  • 2.数组参数和指针参数
    • 2.1一维数组传参
    • 2.2二维数组传参
    • 2.3一级指针传参
    • 2.4二级指针传参
  • 3.函数指针
  • 4.函数指针数组
  • 5.指向函数指针数组的指针
  • 6.回调函数
    • 6.1qsort库函数
      • 6.1.1qsort排序整型数据
      • 6.1.2qsort排序结构体数据
    • 6.2实现冒泡排序对任意类型数据排序

1.字符指针

我们前面已经了解到了,字符指针类型为char*,char*类型的指针是为了存放char类型变量的地址,在解引用时访问一个字节

下面我们来看一下它的另一种使用方式:

#include<stdio.h>
int main()
{char a[] = "happy new year";const char* p = &a;printf("%s\n", p);return 0;
}

这里是把整个字符串放到指针变量p里了吗?答案是否定的,我们有图有真相
这里我们发现p中存放的是a的地址,而a的地址是字符串首元素的也就是字符串中 ‘h’ 的地址,所以上面代码的意思是把一个常量字符串的首字符 h 的地址存放到指针变量 p 中。
结论:

字符指针变量存放字符串地址时存放的是字符串首元素的地址

下面我们再来看一个例题:

#include <stdio.h>
int main()
{char str1[] = "happy new year.";char str2[] = "happy new year";const char* str3 = "happy new year";const char* str4 = "happy new year";if (str1 == str2)printf("str1 and str2 are same\n");elseprintf("str1 and str2 are not same\n");if (str3 == str4)printf("str3 and str4 are same\n");elseprintf("str3 and str4 are not same\n");return 0;
}
//str1and str2 are not same
//str3and str4 are same

这里为什么会输出这个结果呢?原因是这样的:
这里str3和str4指向的是一个同一个常量字符串。C/C++会把常量字符串存储到单独的一个内存区域,当几个指针指向同一个字符串的时候,他们实际会指向同一块内存。但是用相同的常量字符串去初始化不同的数组的时候就会开辟出不同的内存块。所以str1和str2不同,str3和str4不同。

2.指针数组和数组指针

2.1指针数组

指针数组是数组,是个存放指针的数组

这些在前面前面C语言指针详解(上)我们已经讲过了,这里我再举个栗子加深印象

#include<stdio.h>
int main()
{char* arr[3] = { "zhangsan","lisi","wangwu" };int i = 0;for (i = 0; i < 3; i++){printf("%s\n", arr[i]);}return 0;
}
//zhangsan
//lisi
//wangwu

1.2数组指针

数组指针是指针,是能够指向数组的指针。
例如:int(* p)[10]就是一个数组指针,p先和*结合,说明p是一个指针变量,指向的是一个大小为10个整型的数组

用数组指针打印二维数组

#include<stdio.h>
void print(int(*p)[5], int r, int c)
{int i = 0;for (i = 0; i < r; i++){int j = 0;for (j = 0; j < c; 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} };print(arr, 3, 5);//数组名arr,表示首元素的地址//但是二维数组的首元素是二维数组的第一行//所以这里传递的arr,其实相当于第一行的地址,是一维数组的地址,可以数组指针来接收return 0;
}

我们再来看一下int (* p[10])[5]是什么意思
这里我们将其拆开,就成了int(*)[5]和p[10],所以p就是存放5个数组指针的数组

1.3&数组名和数组名

我们知道数组名就是数组首元素的地址,那么&数组名的本质有是什么呢?来看个图:
这里虽然arr和&arr的值是一样的,但是意义是不同的,arr的类型是int*,是一个整型指针类型,+1跳过一个整型,所以 arr+1 相对于 arr 的差值是4,&arr 的类型是 int(*)[10] ,是一种数组指针类型数组的地址+1,跳过整个数组的大小,所以 &arr+1 相对于 &arr 的差值是40
结论:

&arr 表示的是数组的地址,而不是数组首元素的地址

2.数组参数和指针参数

2.1一维数组传参

举起两个栗子:

int arr[10] = {0},可以用int arr[10],int arr[],int* arr来接收

int* arr[10] = {0},可以用int* arr[10],int** arr来接收

2.2二维数组传参

举起一个栗子:

int arr[3][5] = {0},可以用int arr[3][5],int arr[][5],int (*arr)[5]接收

2.3一级指针传参

#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};int *p = arr;int sz = sizeof(arr)/sizeof(arr[0]);//一级指针p,传给函数print(p, sz);return 0;
}

void test1(int *p){},这里的int *p接收的参数为:&变量,数组名,指针

2.4二级指针传参

#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;
}

void test(int** p){},这里的int** p接收的参数为:二级指针,&一级指针,指针数组

3.函数指针

我么们先来看一段代码图:

这里我们打印的是函数Add的地址,观察发现两个地址是一样的
实际上,函数名就是函数的地址,和数组不一样,Add和&Add都是函数的地址,没有区别
要存放函数的地址就需要函数指针了,拿存放函数Add为例

void(* pf)(int x,int y) 就是函数指针,这里void(*)(int x,int y)就是函数指针类型,指向函数,指向的函数参数为int x和int y,返回值为void

我们运用函数指针来将改造一下上面的代码:

void Add(int x, int y)
{return x + y;
}
int main()
{//int(*pf)(int x, int y) = Add;int(*pf)(int,int) = Add;int sum = pf(3, 3);printf("%d", sum);return 0;
}
//输出6

我们再来看看下面两句代码是什么意思:

(* (void ()())0)();
这里是把0当做一个函数的地址,将0直接转换成一个void (
)()的函数指针,然后去调用0地址处的函数

void (* signal(int , void(*)(int)))(int);
上述代码是一次函数声明
声明的函数是signal
signal函数的第一个参数是int类型的
signal函数的第二个参数是一个指针函数类型,该函数指针指向的函数参数是int,返回类型是void
signal函数的返回类型也是一个函数指针类型,该函数指针指向的函数参数是int,返回类型是void

4.函数指针数组

要把函数的地址存到一个数组中,那这个数组就叫函数指针数组
函数指针数组的定义:例如,int (*p[10])(int x,int y);这里p是个有十个元素的数组,p中存放的内容就是函数指针。

我们运用函数指针来写一个计算器:

#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.mui        4.div ******\n");printf("***********  0.exit  ***********\n");printf("********************************\n");
}int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;//函数指针数组 - 转移表int (*pfArr[])(int, int) = { 0,Add,Sub,Mul,Div };do{menu();printf("请选择:>");scanf("%d", &input);if (0 == input){printf("退出计算器\n");break;}if (input >= 1 && input <= 4){printf("请输入两个操作数:>");scanf("%d %d", &x, &y);ret = pfArr[input](x, y);printf("%d\n", ret);}else{printf("选择错误,请重新选择:>\n");}} while (input);return 0;
}

5.指向函数指针数组的指针

指向函数指针数组的指针是一个指针,指针指向一个 数组 ,数组的元素都是函数指针

定义:

#include<stdio.h>
int Add(int x, int y)
{return x + y;
}
int main()
{//函数指针pfvoid (*pf)(const char*) = Add;//函数指针的数组pfArrvoid (*pfArr[5])(const char* str);pfArr[0] = Add;//指向函数指针数组pfunArr的指针ppfunArrvoid (*(*ppfArr)[5])(const char*) = &pfArr;return 0;
}

6.回调函数

回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。

我们运用回调函数来对上面的计算器进行改造:

回调函数
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");
}void calc(int (*p)(int, int))
{int x = 0;int y = 0;int ret = 0;printf("请输入2个操作数:>");scanf("%d %d", &x, &y);ret = p(x, y);printf("%d\n", ret);
}
int main()
{int input = 0;do{menu();printf("请选择:>");scanf("%d", &input);switch (input){case 1:calc(Add);break;case 2:calc(Sub);break;case 3:calc(Mul);break;case 4:calc(Div);break;case 0:printf("退出计算器\n");break;default:printf("输入错误,请重新输入:>\n");break;}} while (input);return 0;
}

6.1qsort库函数


函数讲解:

函数功能:排序任意类型的数据
头文件:#include<stdlib.h>
void* base:待排序数据的起始地址
size_t num:待排序数据的元素个数
size_t size:待排序数据元素的大小(单位是字节)
int (* cmpar)(const void*, const void*) :比较2个元素大小的函数指针
返回值:>0,,p1指向的元素在p2指向的元素之前;=0,p1指向的元素和p2指向的元素相同;<0,p1指向的元素在p2指向的元素之后

6.1.1qsort排序整型数据

//qsort排序整型数组
#include<stdio.h>
int cmp_int(const void* e1, const void* e2)
{return *(int*)e1 - *(int*)e2;
}
int main()
{int arr[10] = { 6,4,3,9,1,5,0,2,7,3 };int sz = sizeof(arr) / sizeof(arr[0]);qsort(arr, sz, sizeof(arr[0]), cmp_int);int i = 0;for (i = 0; i < sz; i++){printf("%d ", arr[i]);}return 0;
}
//0 1 2 3 4 5 6 7 8 9

6.1.2qsort排序结构体数据

按名字来排序

按名字来排序
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
struct Stu
{char name[20];int age;
};
//按名字比较
int cmp_by_name(const void* e1, const void* e2)
{return strcmp(((struct Stu*)e1)->name, ((struct Stu*)e2)->name);
}int main()
{struct Stu s[] = { {"zhangsan",18},{"lisi",30},{"wangwu",24} };int sz = sizeof(s) / sizeof(s[0]);qsort(s, sz, sizeof(s[0]), cmp_by_name);int i = 0;for (i = 0; i < sz; i++){printf("%s %d\n", s[i].name, s[i].age);}return 0;
}//打印输出
//lisi 30
//wangwu 24
//zhangsan 18

按年龄来比较

//按年龄来排序
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
struct Stu
{char name[20];int age;
};
//按年龄比较
int cmp_by_age(const void* e1, const void* e2)
{return  (((struct Stu*)e1)->age - ((struct Stu*)e2)->age);
}
int main()
{struct Stu s[] = { {"zhangsan",18},{"lisi",30},{"wangwu",24} };int sz = sizeof(s) / sizeof(s[0]);qsort(s, sz, sizeof(s[0]), cmp_by_age);int i = 0;for (i = 0; i < sz; i++){printf("%s %d\n", s[i].name, s[i].age);}return 0;
}//打印输出
//zhangsan 18
//wangwu 24
//lisi 30

看到这想必大家对qsort库函数的功能和使用都已经理解了,我们之前讲了冒泡排序,但是冒泡排序只能排序整型数据,下面我们就模仿qsort库函数, 实现用冒泡排序对任意类型的数据排序

6.2实现冒泡排序对任意类型数据排序

代码实现:

//实现冒泡排序对任意类型数据排序
void Swap(char* buf1, char* buf2, int width)
{//挨个儿字节交换int i = 0;for (i = 0; i < width; i++){char tmp = *buf1;*buf1 = *buf2;*buf2 = tmp;buf1++;buf2++;}
}void bubble_sort2(void* base, int sz, int width, int(*cmp)(const void* e1, const void* e2))
{int i = 0;//一趟冒泡排序for (i = 0; i < sz - 1; i++){int j = 0;//每趟交换的次数for (j = 0; j < sz - 1 - i; j++){if (cmp((char*)base + j * width, (char*)base + (j + 1) * width) > 0){//交换Swap((char*)base + j * width, (char*)base + (j + 1) * width, width);}}}
}

我们来测试一下功能是否和我们预期的相同
排序整型数据:

void Swap(char* buf1, char* buf2, int width)
{//挨个儿字节交换int i = 0;for (i = 0; i < width; i++){char tmp = *buf1;*buf1 = *buf2;*buf2 = tmp;buf1++;buf2++;}
}void bubble_sort2(void* base, int sz, int width, int(*cmp)(const void* e1, const void* e2))
{int i = 0;//一趟冒泡排序for (i = 0; i < sz - 1; i++){int j = 0;//每趟交换的次数for (j = 0; j < sz - 1 - i; j++){if (cmp((char*)base + j * width, (char*)base + (j + 1) * width) > 0){//交换Swap((char*)base + j * width, (char*)base + (j + 1) * width, width);}}}
}int cmp_int(const void* e1, const void* e2)
{return *(int*)e1 - *(int*)e2;
}
int main()
{int arr[10] = { 6,4,3,9,1,5,0,2,7,3 };int sz = sizeof(arr) / sizeof(arr[0]);bubble_sort2(arr, sz, sizeof(arr[0]), cmp_int);int i = 0;for (i = 0; i < sz; i++){printf("%d ", arr[i]);}return 0;
}
//0 1 2 3 4 5 6 7 8 9

对结构体排序:
按姓名来排序

void Swap(char* buf1, char* buf2, int width)
{//挨个儿字节交换int i = 0;for (i = 0; i < width; i++){char tmp = *buf1;*buf1 = *buf2;*buf2 = tmp;buf1++;buf2++;}
}void bubble_sort2(void* base, int sz, int width, int(*cmp)(const void* e1, const void* e2))
{int i = 0;//一趟冒泡排序for (i = 0; i < sz - 1; i++){int j = 0;//每趟交换的次数for (j = 0; j < sz - 1 - i; j++){if (cmp((char*)base + j * width, (char*)base + (j + 1) * width) > 0){//交换Swap((char*)base + j * width, (char*)base + (j + 1) * width, width);}}}
}struct Stu
{char name[20];int age;
};
//按名字比较
int cmp_by_name(const void* e1, const void* e2)
{return strcmp(((struct Stu*)e1)->name, ((struct Stu*)e2)->name);
}int main()
{struct Stu s[] = { {"zhangsan",18},{"lisi",30},{"wangwu",24} };int sz = sizeof(s) / sizeof(s[0]);bubble_sort2(s, sz, sizeof(s[0]), cmp_by_name);
int i = 0;
for (i = 0; i < sz; i++)
{printf("%s %d\n", s[i].name, s[i].age);
}return 0;
}//打印输出
//lisi 30
//wangwu 24
//zhangsan 18

按年龄来排序

void Swap(char* buf1, char* buf2, int width)
{//挨个儿字节交换int i = 0;for (i = 0; i < width; i++){char tmp = *buf1;*buf1 = *buf2;*buf2 = tmp;buf1++;buf2++;}
}void bubble_sort2(void* base, int sz, int width, int(*cmp)(const void* e1, const void* e2))
{int i = 0;//一趟冒泡排序for (i = 0; i < sz - 1; i++){int j = 0;//每趟交换的次数for (j = 0; j < sz - 1 - i; j++){if (cmp((char*)base + j * width, (char*)base + (j + 1) * width) > 0){//交换Swap((char*)base + j * width, (char*)base + (j + 1) * width, width);}}}
}struct Stu
{char name[20];int age;
};
//按年龄比较
int cmp_by_age(const void* e1, const void* e2)
{return  (((struct Stu*)e1)->age - ((struct Stu*)e2)->age);
}
int main()
{struct Stu s[] = { {"zhangsan",18},{"lisi",30},{"wangwu",24} };int sz = sizeof(s) / sizeof(s[0]);bubble_sort2(s, sz, sizeof(s[0]), cmp_by_age);int i = 0;for (i = 0; i < sz; i++){printf("%s %d\n", s[i].name, s[i].age);}return 0;
}//打印输出
//zhangsan 18
//wangwu 24
//lisi 30

由上面的用例可见对冒泡排序的改造达到了我们目标要求
好了这次的内容到这里就结束了,请友友们慢慢品味,三连关注不迷路,后期会持续更新C语言干货!

C语言指针详解(下)相关推荐

  1. c语言的指针详解ppt,最全的C语言指针详解.ppt

    最全的C语言指针详解.ppt 第6章 指针,6.1 指针定义与使用 6.2 指针与函数 6.3 指针与数组 6.4 指针与字符串 6.5 指针数组与多级指针 6.6 指针与动态内存分配 6.7 指针的 ...

  2. C语言指针详解(全解)—— 前篇

    关于指针的基本概念及相关运算写在了上一篇博客 C语言指针详解(初级)_Seinrich的博客-CSDN博客 本篇博客来详细的解说一下指针,深入探讨指针 一.指针与const const修饰的变量为常变 ...

  3. C语言指针详解——入门C语言指针,初级指针使用。

    原文地址:我的个人博客点击查看 C语言教程第二弹--指针详解 所有学过C语言的人都知道,C语言难在它的指针的使用和理解,今天,我带给大家C语言的指针的教程,供大家学习交流,如果有讲的不对的地方,请给作 ...

  4. c语言 指针_C 语言指针详解

    (给CPP开发者加星标,提升C/C++技能) 作者:C语言与CPP编程 / 自成一派123(本文来自作者投稿) 1为什么使用指针 假如我们定义了 char a='A' ,当需要使用 'A' 时,除了直 ...

  5. c语言 指针函数 详解,[NOTE-C]C语言指针详解(一)

    C语言指针让一切想法变成可能,强转和指针可以看做一项呼风唤雨的利器,但是C语言中指针应用又需要格外的小心,其更灵活的利用内存,因为不当的应用可能引起各种异常,这篇文章就是让我们一起来认识C指针,更好的 ...

  6. c语言指针详解(概念示例)

    指针是C语言中广泛使用的一种数据类型. 运用指针编程是C语言最主要的风格之一.利用指针变量可以表示各种数据结构: 能很方便地使用数组和字符串: 并能象汇编语言一样处理内存地址,从而编出精练而高效的程序 ...

  7. c语言高低位拷贝_C语言指针详解

    1为什么使用指针 假如我们定义了 char a='A' ,当需要使用 'A' 时,除了直接调用变量 a ,还可以定义 char *p=&a ,调用 a 的地址,即指向 a 的指针 p ,变量 ...

  8. 导航编程用c语言还是c加加,C语言/C加加大神程序员老司机带你玩转C语言指针详解...

    很多初学编程的小伙伴都会选择C语言作为第一门学习的编程语言,因为C语言作为一门底层基础语言相对于其他的高层语言来说更加容易学习.可以来帮助正在学习编程的小伙伴更加快速的了解计算机原理. 但是初学C语言 ...

  9. C语言指针详解(新手入门推荐)

    目录 指针的理解 字符指针:char * 无类型指针void * 指针数组 数组指针 函数指针 回调函数 函数指针数组 练习 指针和数组笔试题解析 指针的理解 关于指针,我先讲一个故事:一个侦探在案发 ...

  10. C语言指针详解——后篇

    目录 一.指针与数组 ​编辑 二.数组传参 2.1一维数组传参 2.2二维数组传参 三.函数指针及应用 四.函数指针数组 一.指针与数组 先来看以下代码及运行的结果 #include <stdi ...

最新文章

  1. 《树莓派Python编程指南》—— 1.3 树莓派快速指南
  2. EID:宏基因组测序在新发腹泻病毒鉴定中的应用
  3. 成功解决ValueError: The truth value of an array with more than one element is ambiguous. Use a.any() or
  4. [密码学] ElGamal加密算法与离散对数
  5. html加注算法源码,200种加密算法(源码)
  6. java 进度条_进度条Java
  7. ffmpeg输出yuv的函数堆栈(h264)
  8. VMware新建虚拟机
  9. 基于DenseNet和自注意机制融合的脐橙病虫害鉴定(DenseNet加入注意力+自然数据集扩大)
  10. 【译】用JavaScript写一个区块链
  11. python贪心算法几个经典例子_python 贪心算法的实现
  12. 如何学好Linux内核?
  13. Android 关于 ActionBarSherlock 的使用
  14. 中软酒店管理系统CSHIS操作手册_数据结构_数据字典
  15. 机器学习分类模型评价指标之混淆矩阵
  16. IT、TT、TN系统,你真的了解吗?
  17. 分享使用PHP开发留言板
  18. JS使用canvas实现(下雨天)特效
  19. response.text和response.content
  20. 生活中我们如何增加多巴胺的分泌

热门文章

  1. SA-A12桌面多媒体电脑音箱,用优质音乐追求精致、品味的生活
  2. 数据库原理及应用复习资料
  3. python要求所有浮点数必须带有小数部分_第2章 基本数据类型
  4. argparse.ArgumentParser()的使用方法
  5. Python爬虫可以赚钱吗
  6. 抓取网页数据做漂亮的图表
  7. 4个WinRAR使用技巧
  8. 发作性睡病的主要症状是什么?
  9. Java 阶段三 Day13 RESTful、Lombok基础组件及Knife4j
  10. 日常维护压力传感器有哪些方法?-道合顺大数据Infinigo