C语言回调函数_judyge的博客-程序员宝宝

1.回调函数与普通函数的区别

从概念上讲,回调函数与普通函数的本质在于:调用者的不同。普通函数由程序员代码调用,而回调函数由操作系统在适当的时间调用。 回调函数主要用于处各种事件和处理。由于WINDOWS系统中存在大量程序员事先不可知的事件,例如鼠标的单击,程序员事先无法得知终端用户何时会发出此动作,因此只能: 
A。定义事件的处理逻辑,与普通函数的编程一样; 
B。告之操作系统自己的处理逻辑,即通知操作系统函数指针;VC/VB等现代编程语言通过事件编程机制隐藏了这一步; 
C。操作系统在事件出现时,调用指定的函数(回调函数的概念)处理,这一步完全由系统负责。

    回调函数在各种操作系统中普遍存在,是现代操作系统为程序员提供处理异步事件的基本机制之一,在不同的系统中的具体实现方式各不相同;请参阅随机文档。Callback 函数实质就是你实现这个函数,由操作系统调用。而一般的情况下是,操作系统提供函数由你来调用的。

2.回调函数实际上就起到了消息循环的作用

    因为在sdk中只有通过回调函数来发送各自的处理消息.

3.C/C++实现

    象C/C++这样支持函数指针的语言都有回调函数的概念,它实际上是向被调用函数传一个你的函数地址,然后被调用函数向通过你传入的函数地址来调用你的函数。比如你做了一个遍历树的函数,但你不知遍历者将对各节点做何种处理时,你就可以在这个遍历函数中加一个函数地址的参数,这样调用者在遍历该树时就可以做各种有意义的工作了:比如打印各节点数据、汇总所有节点之类。

4.Windows回调函数

    回调函数是用来处理窗口消息的函数,一般类型为:

WindowProc(HWND hWnd,UINT message, WPARAM wParam, LPARAM lParam);

hWnd为窗口句柄,message为消息ID,后面两个为消息参数。

    MFC将一部分处理消息的函数封状在CWnd类中,如OnCreate等,其参数也从WPARAM wParam, LPARAM lParam转换为LPCREATESTRUCT结构(可以查看映射宏定义及MFC源代码)而其他的有些也可以用回调函数,如WM_TIMER消息,可以在SetTimer函数里面第三个参数指定回调函数,若为NULL则应该在OnTimer函数中处理改消息

5.MSDN中的描述

Used to asynchronously read the messages in a queue. It is an application-defined function that MSMQ calls when a message is available, a time-out occurs, or an error occurs.

6.Callback最本质的特征包括两点:注册和触发

    Callback函数是你提供给系统调用的函数。很多情况下,系统某个情况下,定义需要执行某个操作,而操作本身由有用户的程序来提供,这时,就要用到回调函数了。所以,简单地说。回调函数,就是你写一个函数,在系统定义的地点提供给系统调用。

    举个例子:SetTimer(),一种处理是,你响应WM_TIMER消息,这暂且不讨论;还有一种用法,就是你提供一个函数,让系统在产生timer消息时自动调用,这种情况下,你可以写好一个timer消息的处理函数,把函数的地址作为SetTimer()的参数,而你这个timer消息的处理函数,就是回调函数。

    其实callback并不仅限于系统调用,用户根据需要,可以建立自己的Callback机制。比如网络通讯,当接收线程(可能专门有一个类封装网络接收行为)收到数据包,需要通知上层(可能又有一个类封装上层数据处理).那么我认为Callback最本质的特征包括两点:注册和触发。实现可以是各种各样的形式,但机制都是如此。比如对于两个类而言,给出以下示例代码:

》》》建立自己的Callback机制

    Callback函数有点类似虚函数,不仅仅系统调用,而且你自己也可以定义Callback函数,比如在自己的类中定义Callback函数的原型,然后在类的其他成员函数中就可以直接调用该Callback函数,而不用管他的具体实现,当然你可以传入参数。而具体实现可能在其他应用程序中或者Dll中,这样可以把接口和实现分离。



以前在某公司实习的时候还说过C语言的回调函数,现在在这说一下。

本代码和语言参考 李先静《系统程序员成长计划》。

回调函数就是由内部实现统一接口,由调用者来决定调用哪一个函数,是对C语言函数指针的一个高级应用。比如我们在Linux内核里面,在设备驱动里面,例如有两个不一样的字符设备。那么对这个设备执行刷新操作(不管是否有刷新操作,这里只举例)需要调用一定的函数,这两个设备所执行的函数一定是不同的。但是,上层的管理数据结构是一样的,在管理数据结构里面分别存入两个刷新操作的函数指针,那么在调用的时候就可以直接用统一接口来执行刷新操作即可。这就是回调函数的一个例子,他实现了传说中的Write Once,Run Anywhere.同时维护起来跟其他的相比更简单。

那么就看看书上的例子:实现一个双向链表(通用链表),它有遍历链表、统计链表中最大值和链表值求和的功能。这三个功能可以写成三个不同也不相关的函数,但是我们可以想到,这三个函数都有着相同的操作,那就是依次遍历。那么就可以将这个遍历抽出来(代码中的dlist_foreach函数),然后根据这三个函数的各自的功能来实现较少重复的代码。 具体代码请看下面:

? Download d_list.c
#include<stdio.h>
#include<stdlib.h>
#include<malloc.h>
 
typedef struct _DListNode{
        
	struct _DListNode *prev;
	struct _DListNode *next;
	void *data;
}DListNode;
 
typedef void (*DListFunc)(void *ctx,void *data);
 
 
void add_dlist_node_int(DListNode *root,DListNode *p){
        
	if(root==NULL)return;
	if(p==NULL)return;
	p->next=root->next;
	p->prev=root;
	root->next=p;
}
 
void delete_dlist_node(DListNode *d){
        
	if(NULL==d)return;
	print_int(NULL,d);
	d->prev->next=d->next;
	d->next->prev=d->prev;
	free(d->data);
	free(d);
	d=NULL;
}
 
void init(DListNode **root){
        
	int i=0;
	int *tmp=NULL;
	*root=(DListNode*)(malloc(sizeof(DListNode)));
	(*root)->data=(int *)(malloc(sizeof(int)));
	*((int *)(*root)->data)=-1;
	(*root)->next=(*root)->prev=*root;
	for(i=0;i<10;i++){
        
		DListNode *p=(DListNode *)(malloc(sizeof(DListNode)));
		tmp=(int *)(malloc(sizeof(int)));
		*tmp=i;
		p->data=tmp;
		add_dlist_node_int(*root,p);
	}
}
 
void dlist_foreach(DListNode *thiz,DListFunc visit,void *context){
        
	DListNode *iter=NULL;
	if(NULL==thiz)return;
	iter=thiz->next;
	while(iter!=NULL&&iter!=thiz){
        
		visit(context,iter);
		iter=iter->next;
	}
}
 
void print_int(void *thiz,DListNode *d){
        
	printf("======%d\n",*((int *)(d->data)));
}
 
void sum_cb(void *thiz,DListNode *d){
        
	(*(int *)(thiz))+=*((int *)(d->data));
}
 
void find_max(void *thiz,DListNode *d){
        
	int a1=*((int *)(thiz));
	int a2=*((int *)(d->data));
	if(a1<a2){
        
		*((int *)thiz)=a2;
	}
}
 
void destroy(DListNode **root){
        
	DListNode *iter1=NULL,*iter2=NULL;
	if(NULL==(*root))return;
	iter1=*root;
	while(iter1!=NULL){
        
		iter2=iter1->next;
		if(iter2==iter1)iter2=NULL;
		delete_dlist_node(iter1);
		iter1=iter2;
	}
}
 
int main()
{
        
	DListNode *root=NULL;
	int tmp=0;
	init(&root);
	if(root==NULL){
        
		printf("Root id NULL!\n");
	}
	dlist_foreach(root,print_int,&tmp);
	tmp=0;
	dlist_foreach(root,sum_cb,&tmp);
	printf("SUB IS: %d\n",tmp);
	tmp=0;
	dlist_foreach(root,find_max,&tmp);
	printf("MAX IS: %d\n",tmp);
 
	destroy(&root);
}
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/judyge/article/details/41281829

智能推荐

vue项目实战(三)- 旅游网站详情页面开发_i小默的博客-程序员宝宝_vue 网站

一、详情页banner添加动态路由:{ path: '/detail/:id', name: 'Detail', component: Detail }详情页首图:&lt;template&gt; &lt;div class="banner"&gt; &lt;img class="banner-img" src="//img1.qunarzz.com/sight/p0/1809/10/10d6568ad3ad4bb5a3.img.jpg

husky_gazebo没有/scan_贤贤贤贤小波的博客-程序员宝宝

系统:ubuntu 18.04ros:melodic启动仿真环境roslaunch husky_gazebo husky_playpen.launch安装的ros和husky都正常,但是偏偏没有/scan数据排查了很久后来发现目前版本是21年7月份更新的 0.4.10版本发现之前安装过的 0.4.8的版本是正常的,把旧版本替换新版本即可原本想通过apt安装旧版本,执行 apt-cache madison husky 发现只有0.4.10单版本,没有历史版本,换过 阿...

Linux 下的一个全新的性能测量和调式诊断工具 Systemtap, 第 2 部分: DTrace_ctthuangcheng的博客-程序员宝宝

DTrace的原理本系列文章详细地介绍了一个 Linux 下的全新的调式、诊断和性能测量工具 Systemtap 和它所依赖的基础 kprobe 以及促使开发该工具的先驱 DTrace 并给出实际使用例子使读者更进一步了解和认识这些工具。 本文是该系列文章之二,它详细地讲解了 DTrace 的原理。本系列文章之一讲解了 kprobe 的原理、编程接口、局限性和使用注意事项并给出实际使用示例帮助读者

get请求与post请求的特点_........千年老妖的博客-程序员宝宝_get请求特点

GET请求具有以下的几个特点:GET请求可被缓存GET请求保留在浏览器历史记录中GET请求可被收藏为书签GET请求不应在处理敏感数据时使用GET请求有长度限制GET请求只应当用于取回数据POST请求的特点如下:POST请求不会被缓存POST请求不会保留在浏览器历史记录中POST请求不能被收藏为书签POST请求对数据长度没有要求...

Qt5下实现摄像头预览及捕获图像方法二(openCV3与Qt5交互使用)_worthsen的博客-程序员宝宝

致谢:http://www.cnblogs.com/annt/p/ant003.html前言:OpenCV对图像及视频的处理方便且很专业,对于摄像头的支持也很好,但有个不足就是它虽然具有GUI模块(即highgui),但是实在是很简陋,就连一个按键都无法直接实现(需要借助滚动条实现),这一点难以满足可视化的图像处理的想法;另一方面,Qt作为一个优秀的图形库,在GUI上表现出色,且界面设计可

【DX11习题学习二】第六章练习 Drawing in Direct3D(上)_键盘崩坏的博客-程序员宝宝

本系列只针对书中每章节的编程练习题,不涉及书中的数学题,需要数学部分的解答请点击对应原书 P137 4.7EXERCISES1. Write down the D3D10_INPUT_ELEMENT_DESC array for the following vertex structure:struct Vertex{XMFLOAT3 Pos;XMFLO

随便推点

DirectX11 With Windows SDK--11 混合状态_weixin_30528371的博客-程序员宝宝

DirectX11 With Windows SDK--11 混合状态 原文:DirectX11 With Windows SDK--11 混合状态前言这一章会着重讲述混合状态,在下一章则会讲述深度/模板状态DirectX11 With Windows SDK完整目录Github项目源码欢迎加入QQ群: 727623616 可以...

day3-python-集合文件操作函数_weixin_30446197的博客-程序员宝宝

一、集合主要作用:去重关系测试, 交集\差集\并集\反向(对称)差集#!/usr/bin/env python# -*- coding:utf-8 -*-list_1 = [1,4,5,7,3,6,7,9]list_1 = set(list_1)list_2 = set([2,6,0,66,22,8,4])print(list_1,list_2)...

Windows密码复杂性要求_allway2的博客-程序员宝宝_密码必须符合复杂性要求

密码必须符合复杂性要求介绍 "密码必须满足复杂性要求" 安全策略设置的最佳做法、位置、值和安全注意事项。参考"密码必须满足复杂性要求" 策略设置确定密码是否必须满足一系列对强密码重要的指南。 启用此策略设置需要密码才能满足以下要求:在更改或创建密码时, 将强制执行复杂性要求。Windows Server 密码复杂性要求中包含的规则属于 Passfilt, 不能直接修改。启用...

NOIP2002-普及组复赛-第二题-级数求和_weixin_30345577的博客-程序员宝宝

题目描述Description  已知:Sn= 1+1/2+1/3+…+1/n。显然对于任意一个整数K,当n足够大的时候,Sn大于K。  现给出一个整数K(1&lt;=k&lt;=15),要求计算出一个最小的n;使得Sn>K。输入输出格式Input/output输入格式:一个正整数K。输出格式:一个正整数N。...

python练习12_い木乄子゛的博客-程序员宝宝

题目:判断101-200之间有多少个素数,并输出所有素数。 #!/usr/bin/python# -*- coding: UTF-8 -*-#from math import sqrtprime = []flag = Truefor i in range(101, 201): k = int(sqrt(i)) for j in range(2, k + 1): ...

python—异常处理_didaojiao4342的博客-程序员宝宝

Python 之-异常处理异常和错误part1:程序中难免出现错误,而错误分成两种1语法错误(这种错误,根本过不了Python解释器的语法检测,必须在程序执行前就改正)#语法错误示范一、if#语法错误示范二、def test: pass#语法错误示范三、print(haha2逻辑错误(逻辑错误)#用户输入...

推荐文章

热门文章

相关标签