手把手教你玩转网络编程模型之完成例程(Completion Routine)篇(上)

手把手教你玩转网络编程模型之完成例程(Completion Routine)

  

记得写这个系列的上一篇文章的时候已经是四年前了,准确的说是四年半以前了,翻开我尘封已久的IO模型里面的“完成例程”的实现方式及示例代码。

本文凝聚着笔者心血,如要转载,请指明原作者及出处,谢谢!不过代码写得不好,欢迎改进,而且没有版权,请随便散播、使用。OK, Let’s go ! Have fun 

本文配套的示例源码下载地址(在我的下载空间里)

http://piggyxp.download.csdn.net/

MFC代码,配有非常非常详尽的注释,功能只是简单的显示一下各个客户端发来的字符,作为教学代码,为了使得代码结构清晰明了,简化了很多地方,用于产品开发的话还需要做很多改进,有错误或者不足的地方,非常欢迎大家不吝指出。完成例程模型示例

 

本文假设你已经对重叠I/O的机制已有了解,否则请先参考本系列的前一篇把手教你玩转重叠IO模型》

 

目录:

1.完成例程的优点

2.完成例程的基本原理

3.关于完成例程的函数介绍

4.完成例程的实现步骤

5.实际应用中应该进一步完善的地方

 

 

一.        完成例程的优点

1.   首先需要指明的是,这里的“完成例程”(Completion Routine)并非是大家所常听到的“完成端口”(Completion Port),而是另外一种管理重叠I/O请求的方式,而至于什么是重叠I/O,简单来讲就是Windows系统内部管理I/O的一种方式,核心就是调用的ReadFileWriteFile函数,在制定设备上执行I/O操作,不光是可用于网络通信,也可以用于其他需要的地方。

Windows系统中,管理重叠I/O可以有三种方式:

(1) 上一篇中提到的基于事件通知的重叠I/O模型

 (2) 本篇中将要讲述的基于“完成例程”的重叠I/O模型

 (3) 下一篇中将要讲到的“完成端口”模型

虽然都是基于重叠I/O,但是因为前两种模型都是需要自己来管理任务的分派,所以性能上没有区别,而完成端口是创建完成端口对象使操作系统亲自来管理任务的分派,所以完成端口肯定是能获得最好的性能。

2.   如果你想要使用重叠I/O机制带来的高性能模型,又懊恼于基于事件通知的重叠模型要收到64个等待事件的限制,还有点畏惧完成端口稍显复杂的初始化过程,那么“完成例程”无疑是你最好的选择!^_^因为完成例程摆脱了事件通知的限制,可以连入任意数量客户端而不用另开线程,也就是说只用很简单的一些代码就可以利用Windows内部的I/O机制来获得网络服务器的高性能,是不是心动了呢?那就一起往下看。。。。。。。。。。

3.   而且个人感觉“完成例程”的方式比重叠I/O更好理解,因为就和我们传统的“回调函数”是一样的,也更容易使用一些,推荐!

 

二.        完成例程的基本原理

概括一点说,上一篇拙作中提到的那个基于事件通知的重叠I/O模型,在你投递了一个请求以后(比如WSARecv),系统在完成以后是用事件来通知你的,而在完成例程中,系统在网络操作完成以后会自动调用你提供的回调函数,区别仅此而已,是不是很简单呢?

首先这里统一几个名词,包括“重叠操作”、“重叠请求”、“投递请求”等等,这是为了配合这的重叠I/O才这么讲的,说的直白一些,也就是你在代码中发出的WSARecv()WSASend()等等网络函数调用。

 上篇文章中偷懒没画图,这次还是画个流程图来说明吧,采用完成例程的服务器端,通信流程简单的来讲是这样的:

 

 完成例程流程图

 

从图中可以看到,服务器端存在一个明显的异步过程,也就是说我们把客户端连入的SOCKET与一个重叠结构绑定之后,便可以将通讯过程全权交给系统内部自己去帮我们调度处理了,我们在主线程中就可以去做其他的事情,边等候系统完成的通知就OK,这也就是完成例程高性能的原因所在。

如果还没有看明白,我们打个通俗易懂的比方,完成例程的处理过程,也就像我们告诉系统,说“我想要在网络上接收网络数据,你去帮我办一下”(投递WSARecv操作),“不过我并不知道网络数据合适到达,总之在接收到网络数据之后,你直接就调用我给你的这个函数(比如_CompletionProess),把他们保存到内存中或是显示到界面中等等,全权交给你处理了”,于是乎,系统在接收到网络数据之后,一方面系统会给我们一个通知,另外同时系统也会自动调用我们事先准备好的回调函数,就不需要我们自己操心了。

看到这里,各位应该已经对完成例程的体系结构有了比价清晰的了解了吧,下面各位喝点咖啡转转脖子休息休息,然后就进入到下面的具体实现部分了。

 

一.        完成例程的函数介绍

这个部分将要介绍在完成例程模型中会使用到的关键函数,内容比较枯燥,大家要做好心理准备。不过在实际应用以前,很多东西肯定也不会理解得太深刻,可以先泛泛的了解一下,以后再回头复习这里的知识就可以了。

厄。。。。。。仔细审查了一下代码,发现其实这里也没有什么新函数好介绍了,大部分都是使用重叠模型那一章里介绍的一样的函数,需要查看的朋友请看这里《手把手教你玩转重叠IO模型》

,这里就不再重复了:

这里只补充一个知识点,就是咱们完成例程方式和前面的事件通知方式最大的不同之处就在于,我们需要提供一个回调函数供系统收到网络数据后自动调用,回调函数的参数定义应该遵照如下的函数原型:

 

1. 完成例程回调函数原型及传递方式

函数应该是这样定义的,函数名字随便起,但是参数类型不能错

 

      

还有一点需要重点提一下的是,因为我们需要给系统提供一个如上面定义的那样的回调函数,以便系统在完成了网络操作后自动调用,这里就需要提一下究竟是如何把这个函数与系统内部绑定的呢?如下所示,在WSARecv函数中是这样绑定的

其他参数我们可以先不用先细看,只看最后一个,看到了吗?直接在

MSDN这么个好帮手,而且在讲后面的完成例程和完成端口的时候我还会讲到一些<FONT color="blue" face="""> ^_^

/*********** TM1680 参考程序: 1、A1\A0 采用MCU进行控制,实际使用时,可以将A1\A0任意接高低电平,TM1680 ID改为相应指令即可; 2、该程序采用STC15W 芯片模拟IIC协议,IO口为双向IO口(无需设置输入与输出),如果MCU的IO口需要设置输入和输出,则在ACK时需要设置为输入 3、该芯片支持标准IIC协议 ************/ #include #include "intrins.h" #include /****命令宏定义****/ #define TM1680ID 0xe7 #define SYSDIS 0x80 #define SYSEN 0x81 #define LEDOFF 0x82 #define LEDON 0x83 #define BLINKOFF 0x88 #define BLINK2HZ 0x89 #define BLINK1HZ 0x8A #define BLINK0_5HZ 0x8B #define SLAVEMODE 0x90 #define RCMODE0 0x98 #define RCMODE1 0x9A #define EXTCLK0 0x9C #define EXTCLK1 0x9E #define COM8NMOS 0xA0 #define COM16NMOS 0xA4 #define COM8PMOS 0xA8 #define COM16PMOS 0xAC #define PWM01 0xB0 #define PWM02 0xB1 #define PWM03 0xB2 #define PWM04 0xB3 #define PWM05 0xB4 #define PWM06 0xB5 #define PWM07 0xB6 #define PWM08 0xB7 #define PWM09 0xB8 #define PWM10 0xB9 #define PWM11 0xBA #define PWM12 0xBB #define PWM13 0xBC #define PWM14 0xBD #define PWM15 0xBE #define PWM16 0xBF /******命令宏定义******/ /***端口定义***/ sbit SDA=P1^4; //TM1680通讯端口设置 sbit SCL=P1^5; sbit MA1=P1^6; sbit MA0=P1^7; /***按键功能设置***/ sbit KEY0=P3^2; sbit KEY1=P3^3; sbit KEY2=P3^6; /***LED指示灯定义***/ sbit RED=P3^4; sbit GREEN=P3^5; sbit WHITE=P3^7; /***显示数据***/ unsigned char TM1680perseg[8]={0x10,0x20,0x40,0x80,0x01,0x02,0x04,0x8}; unsigned char DispA[8]={0x10,0xFE,0x92,0x92,0xFE,0x92,0x10,0x10}; /*** 函数功能:延时函数 ***/ void delayms(unsigned int n) { unsigned int i; while(n--) { for(i=0;i<550;i++); } } void delayus(unsigned char n) //256 { while (--n) { _nop_(); } } /**************************底层函数*****************************/ void TM1680start(void) { SDA=1; SCL=1; SDA=1; delayus(4); delayus(10); SDA=0; delayus(10); //起始信号,必须大于4.7us SCL=0; } void Ack(void) { SCL = 0; delayus(8); SCL = 1; delayus(8); while(SDA); SCL=0; delayus(15); } void TM1680SetAck(bit ack) { SCL=0; delayus(5); SDA = ack; //写应答信号 SCL = 1; //拉高时钟线 delayus(5); //延时 SCL = 0; //拉低时钟线 delayus(5); //延时 } void TM1680stop(void) { SDA=0; SCL=1; delayus(10); SDA=1; //停止信号,大于5us delayus(10); SCL=1; SDA=1; } void TM1680SendByte(unsigned char sbyte) { unsigned char i=0; for(i=0; i<8; i++) { SCL=0;delayus(2); if(sbyte&0x80;) { SDA=1; //高位先发 }else{ SDA=0; } delayus(3); SCL=1; delayus(5); //高电平的时间大于4us sbyte<<=1; delayus(2); } SCL=0; delayus(3); SDA=0; delayus(3); } unsigned char TM1680RecvByte(void) { unsigned char i=0, sbyte; SDA=1;delayus(6); for(i=0; i<8; i++) { SCL=0;delayus(6); if(SDA) { sbyte |= 0x01 ; //置1 }else{ sbyte &= 0xfe; //置0 } delayus(3); SCL=1; delayus(5); //高电平的时间大于4us sbyte<<=1; } SCL=0; return sbyte; } /******************************底层函数结束**************************/ /******************************功能函数**************************/ /***单字节写操作函数***/ /***写命令函数:开始--ID-ACK--命令-ACK--结束***/ void TM1680WriteCmd(unsigned char scmd) { TM1680start(); TM1680SendByte(TM1680ID); Ack(); TM1680SendByte(scmd); Ack(); TM1680stop(); } /***写一个字节数据: 开始--ID-ACK-内部地址-ACK--数据-ACK-结束 ***/ void TM1680WriteOneByte(unsigned char faddr, unsigned char sdate) { TM1680start(); TM1680SendByte(TM1680ID); //写TM1680器件地址 Ack(); TM1680SendByte(faddr); //eeprom 地址 Ack(); TM1680SendByte(sdate); //写数据 Ack(); TM1680stop(); } /*** 函数功能:页操作 ***/ void TM1680PageWrite(unsigned char faddr, unsigned char *pdate,unsigned char cnt) { unsigned char i=0; TM1680start(); TM1680SendByte(TM1680ID); //写TM1680器件地址 Ack(); TM1680SendByte(faddr); //eeprom 地址 Ack(); for(i=0; i<cnt; i++) { TM1680SendByte(*pdate); //写数据 Ack(); pdate++; } TM1680stop(); } void TM1680PageAllWrite(unsigned char faddr, unsigned char sdate,unsigned char cnt) { unsigned char i=0; TM1680start(); TM1680SendByte(TM1680ID); Ack(); TM1680SendByte(faddr); //eeprom 地址 Ack(); for(i=0; i<cnt; i++) { TM1680SendByte(sdate); //写数据 Ack(); } TM1680stop(); } /*** 函数功能: 写命令+写数据 ***/ void TM1680WriteCmdDate(unsigned char faddr, unsigned char sdate, unsigned char cnt) { unsigned char i=0; TM1680start(); TM1680SendByte(TM1680ID); Ack(); TM1680SendByte(SYSDIS); Ack(); TM1680SendByte(COM8NMOS); Ack(); TM1680SendByte(RCMODE1); Ack(); TM1680SendByte(SYSEN); Ack(); TM1680SendByte(LEDON); Ack(); TM1680SendByte(PWM16); Ack(); TM1680SendByte(BLINKOFF); Ack(); TM1680SendByte(faddr); Ack(); for(i=0; i<cnt; i++) { TM1680SendByte(sdate); Ack(); } TM1680stop(); } /******************************************函数功能结束****************************************/ /*** 函数功能: TM1680 初始化 ***/ void TM1680Init(void) { TM1680start(); TM1680SendByte(TM1680ID); Ack(); TM1680SendByte(SYSDIS); Ack(); TM1680SendByte(COM8NMOS); //根据需求进行选择 Ack(); TM1680SendByte(RCMODE1); Ack(); TM1680SendByte(SYSEN); Ack(); TM1680SendByte(LEDON); Ack(); // TM1680SendByte(LEDOFF); // Ack(); TM1680SendByte(PWM16); Ack(); TM1680SendByte(BLINKOFF); Ack(); TM1680stop(); } /***PWM调节测试***/ void PWMTest(unsigned char sTime) { unsigned char i=0; for(i=0;i<16;i++) { TM1680WriteCmd(PWM01 | i); delayms(sTime); delayms(sTime); } } /***TM1680ID判断***/ void TM1680IDChange(void) { switch(TM1680ID) { case 0xe4: MA1=0; MA0=0; break; case 0xe5: MA1=0; MA0=1; break; case 0xe6: MA1=1; MA0=0; break; case 0xe7: MA1=1; MA0=1; break; default:break; } } /**** 函数功能:逐渐点亮每一段 ****/ void TM1680PerDisp(void) { unsigned char i=0,j=0; unsigned char faddr=0x00, fdate=TM1680perseg[0]; for(i=0;i<32; i++) { for(j=0;j<8;j++) { fdate=fdate|TM1680perseg[j]; TM1680WriteOneByte(faddr,fdate); delayms(20); } faddr+=2; fdate=TM1680perseg[0]; } } void main(void) { unsigned char i=0,j=0,fdate=0x10; TM1680IDChange(); TM1680Init(); TM1680PageAllWrite(0x00, 0x00, 32); //上电清零 delayms(100); TM1680PageAllWrite(0x00, 0xff, 32); //上电全部显示 TM1680PerDisp(); //每一段显示 while(1) { ; } }
评论 27
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值