理解函数指针和回调函数

理解

函数指针

指向函数的指针。比如:

理解函数指针的伪代码
void (*p)(int type, char *data); // 定义一个函数指针p
void func(int type, char *data); // 声明一个函数func
p = func; // 将指针p指向函数func
p(1,"test"); // 调用方式1
(*p)(1,"test"); // 调用方式2

回调函数和回调

比如,B把自己函数cbkFunc()的地址告诉A,A在运行过程中执行cbkFunc()。则
回调函数:指B的函数cbkFunc();
注册回调函数:指B把函数cbkFunc()的地址告诉A;
回调:指A在运行过程中执行cbkFunc()。

code

  • 无头文件.h
  • 为了方便模块A和B写在了一个.c文件中
  • 打印log的函数见参考博客[3],可以直接替换为printf()
FunctionPointTest.c
/*
 * 理解函数指针和回调函数
 *
 * 假设需求:A要进行某项运动,有开始、正在做、结束3个状态,
 * B需要关注这3个状态
 *
 * 实现方案:A提供一个回调函数注册接口,在程序开始运行时,
 * B向A注册回调函数,A以回调函数的形式通知B
 *
 * 为了简便A和B写在一个文件里
 */

#include "D:\MyFiles\MyLog\WindowsC\mylog.h"
#include <stdio.h>
#include <windows.h>

// A和B的共同定义,一般是A的一个头文件,B会包含这个头文件:
typedef enum ENUM_EVENT {
	E_EVENT_START,
	E_EVENT_DOING,
	E_EVENT_FINISH,
} EnumEvent;
typedef void (*EVENT_CBK)(EnumEvent type, char *data);
int regEventCbk(EVENT_CBK cbk);

// A的实现
#define TAGA "[MODULE_A]"
static EVENT_CBK gSendEvent = NULL;
int regEventCbk(EVENT_CBK cbk) {
	if (cbk == NULL) {
		return -1;
	}
	gSendEvent = cbk;
	LOGI("%s{有人注册了回调函数:%X, 地址:%p}", TAGA, *gSendEvent, gSendEvent);
	return 0;
}
void doing() {
	gSendEvent(E_EVENT_DOING, "进行中..."); // 调用方式1
	Sleep(1000);
}
void runA() {
	LOGI("%s{我是A}", TAGA);
	if (gSendEvent == NULL) {
		LOGW("%s{B不关心我}", TAGA);
		return;
	}
	(*gSendEvent)(E_EVENT_START, "我开始了哟"); // 调用方式2
	doing();
	(*gSendEvent)(E_EVENT_FINISH, "我好了");
}

// B的实现
#define TAGB "[MODULE_B]"
long long int gAStart = 0;
long long int gAFinish = 0;
void eventStartHandler(char *data) {
	gAStart = GetTickCount();
	LOGI("%s{A:%s}{%lld}", TAGB, data, gAStart);
}
void eventDoingHandler(char *data) {
	LOGI("%s{A:%s}", TAGB, data);
}
void eventFinishHandler(char *data) {
	long long int aDurS = 0;
	gAFinish = GetTickCount();
	LOGI("%s{A:%s}{%lld}", TAGB, data, gAFinish);

	aDurS = (gAFinish - gAStart)/1000;
	LOGI("%s{A %llds}", TAGB, aDurS);
	if (aDurS < 10) {
		LOGI("%s{A 真快}", TAGB);
	}
}
void onEvent(EnumEvent type, char *data) {
	switch (type) {
		case E_EVENT_START:
			eventStartHandler(data);
		break;
		case E_EVENT_DOING:
			eventDoingHandler(data);
		break;
		case E_EVENT_FINISH:
			eventFinishHandler(data);
		break;
		default:
		break;
	}
}
void careA() {
	LOGI("%s{我的回调函数地址:%p}", TAGB, onEvent);
	regEventCbk(onEvent);
}
void runB() {
	LOGI("%s{我是B}", TAGB);
	careA();
}




int main() {
	LOGI("{start}");
	runB();
	runA();
	return 0;
}

运行结果

image

可优化点

.

参考博客

[1] https://zhuanlan.zhihu.com/p/162578969
[2] https://blog.csdn.net/zhou8201/article/details/100700479
[3] https://blog.csdn.net/qq_31300101/article/details/130190026?spm=1001.2014.3001.5502

原文链接:https://www.cnblogs.com/livetime/p/17335033.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:理解函数指针和回调函数 - Python技术站

(0)
上一篇 2023年4月18日
下一篇 2023年4月22日

相关文章

  • 实例解析Json反序列化之ObjectMapper(自定义实现反序列化方法)

    下面是详细讲解“实例解析Json反序列化之ObjectMapper(自定义实现反序列化方法)”的完整攻略: 1. 什么是Json反序列化? Json反序列化是将Json数据类型转换成Java对象的过程。在Java中,我们通常使用Jackson库来实现Json数据的反序列化。Jackson库中的ObjectMapper类提供了非常强大的反序列化功能,它可以将J…

    C 2023年5月23日
    00
  • C语言实现简单猜拳小游戏

    C语言实现简单猜拳小游戏 简介 猜拳是一种非常简单有趣的游戏,通过手势判断胜负,容易上手,适合初学者练习编程。本文将讲解如何使用C语言实现简单的猜拳小游戏。 程序设计思路 猜拳游戏可以分为以下几个步骤: 程序提示用户输入出拳手势(剪刀、石头、布); 程序随机生成出拳手势; 程序比较用户输入和程序生成的手势,判断胜负,输出结果。 根据以上步骤,我们可以设计以下…

    C 2023年5月23日
    00
  • C 标准库 ctype.h

    ctype.h 是 C 标准库中的一个头文件,提供了一些用于字符处理的函数。这里详细讲解一下它的使用方法。 ctype.h 头文件的引入 为了使用 ctype.h 头文件,需要在程序中包含它。可以使用以下代码引入: #include <ctype.h> 一些常用的 ctype.h 函数 isalnum() 此函数用于检查字符是否是字母或数字。如果…

    C 2023年5月10日
    00
  • C++模拟实现string的示例代码

    以下是“C++模拟实现string的示例代码”的完整攻略。 步骤一:定义头文件 首先要定义一个NameSpace,包含实现string所需的类和函数,然后定义头文件,并把实现代码加入其中。 namespace my_string{ class string; } class my_string::string{ private: char* _data; s…

    C 2023年5月24日
    00
  • __stdcall 和 __cdecl 的区别浅析

    关于“__stdcall 和 __cdecl 的区别浅析”这一话题,下面为你提供一份详细的攻略。 简介 __stdcall 和 __cdecl 是 C++ 中函数调用的两种不同的方式,它们都在函数名后面加上了一些符号来指示参数传递的方式。具体来说: __stdcall:参数从右往左依次压入堆栈,被调用函数从堆栈中获取参数,由被调用函数负责堆栈内存的清理工作。…

    C 2023年5月23日
    00
  • JS将网址url转化为JSON格式的方法

    将网址URL转换为JSON格式是一种常见的操作。下面是一些步骤,可帮助您使用JavaScript在JSON中使用网址: 创建URL对象。 使用JavaScript的URL对象,可以将网址转换为URL对象。例如,下面的代码将创建一个URL对象: const url = new URL(‘https://www.example.com/test/?id=123#…

    C 2023年5月23日
    00
  • c++ 如何合并两个有序链表

    合并两个有序链表是一个经典的算法问题。下面将详细讲解使用C++解决这个问题的完整攻略。 问题描述 合并两个有序链表为一个新的有序链表。 解决思路 迭代法 迭代法的思路是:比较两个链表的节点,将较小的节点加入合并后的链表,直到有一个链表为空。此时将另一个非空链表节点全部加入合并后的链表即可。 递归法 递归法的思路是:比较两个链表的头部,较小的节点加入合并后的链…

    C 2023年5月23日
    00
  • C调用C++代码的方法步骤

    C语言是一款面向过程的语言,而C++语言是一款面向对象的语言。虽然二者有着许多相似之处,但仍然会存在一些不兼容的情况,在C中调用C++代码时也是如此。下面介绍一下C调用C++代码的方法步骤。 步骤 在C++文件中,声明被调用的函数为extern “C” #ifdef __cplusplus extern "C" { #endif // y…

    C 2023年5月23日
    00
合作推广
合作推广
分享本页
返回顶部