`
chanshui
  • 浏览: 83006 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论
阅读更多
对于许多初学者来说,网络通信程序的开发,普遍的一个现象就是觉得难以入手。许多概念,诸如:同步(Sync)/异步(Async),阻塞(Block)/非阻塞(Unblock)等,初学者往往迷惑不清,只知其所以而不知起所以然。

同步方式指的是发送方不等接收方响应,便接着发下个数据包的通信方式;而异步指发送方发出数据后,等收到接收方发回的响应,才发下一个数据包的通信方式。

阻塞套接字是指执行此套接字的网络调用时,直到成功才返回,否则一直阻塞在此网络调用上,比如调用recv()函数读取网络缓冲区中的数据,如果没有数据到达,将一直挂在recv()这个函数调用上,直到读到一些数据,此函数调用才返回;而非阻塞套接字是指执行此套接字的网络调用时,不管是否执行成功,都立即返回。比如调用recv()函数读取网络缓冲区中数据,不管是否读到数据都立即返回,而不会一直挂在此函数调用上。在实际Windows网络通信软件开发中,异步非阻塞套接字是用的最多的。平常所说的C/S(客户端/服务器)结构的软件就是异步非阻塞模式的。

对于这些概念,初学者的理解也许只能似是而非,我将用一个最简单的例子说明异步非阻塞Socket的基本原理和工作机制。目的是让初学者不仅对Socket异步非阻塞的概念有个非常透彻的理解,而且也给他们提供一个用Socket开发网络通信应用程序的快速入门方法。操作系统是Windows 98(或NT4.0),开发工具是Visual C++6.0。

MFC提供了一个异步类CAsyncSocket,它封装了异步、非阻塞Socket的基本功能,用它做常用的网络通信软件很方便。但它屏蔽了Socket的异步、非阻塞等概念,开发人员无需了解异步、非阻塞Socket的原理和工作机制。因此,建议初学者学习编网络通信程序时,暂且不要用MFC提供的类,而先用Winsock2 API,这样有助于对异步、非阻塞Socket编程机制的理解。

为了简单起见,服务器端和客户端的应用程序均是基于MFC的标准对话框,网络通信部分基于Winsock2 API实现。

先做服务器端应用程序。

用MFC向导做一个基于对话框的应用程序SocketSever,注意第三步中不要选上Windwos Sockets选项。在做好工程后,创建一个SeverSock,将它设置为异步非阻塞模式,并为它注册各种网络异步事件,然后与自定义的网络异步事件联系上,最后还要将它设置为监听模式。在自定义的网络异步事件的回调函数中,你可以得到各种网络异步事件,根据它们的类型,做不同的处理。下面将详细介绍如何编写相关代码。

在SocketSeverDlg.h文件的类定义之前增加如下定义:
显示代码
打印
01 #define NETWORK_EVENT WM_USER+166 file://定义网络事件
02
03 SOCKET ServerSock; file://服务器端Socket
04
05 在类定义中增加如下定义:
06
07 class CSocketSeverDlg : CDialog
08
09 {
10
11 …
12
13 public:
14
15 SOCKET ClientSock[CLNT_MAX_NUM]; file://存储与客户端通信的Socket的数组
16
17 /*各种网络异步事件的处理函数*/
18
19 void OnClose(SOCKET CurSock); file://对端Socket断开
20
21 void OnSend(SOCKET CurSock); file://发送网络数据包
22
23 void OnReceive(SOCKET CurSock); file://网络数据包到达
24
25 void OnAccept(SOCKET CurSock); file://客户端连接请求
26
27 BOOL InitNetwork(); file://初始化网络函数
28
29 void OnNetEvent(WPARAM wParam, LPARAM lParam); file://异步事件回调函数
30
31 …
32
33 };





在SocketSeverDlg.cpp文件中增加消息映射,其中OnNetEvent是异步事件回调函数名:
显示代码
打印
1 ON_MESSAGE(NETWORK_EVENT,OnNetEvent)





定义初始化网络函数,在SocketSeverDlg.cpp文件的OnInitDialog()中调此函数即可。
显示代码
打印
001 BOOL CSocketSeverDlg::InitNetwork()
002
003 {
004
005 WSADATA wsaData;
006
007 file://初始化TCP协议
008
009 BOOL ret = WSAStartup(MAKEWORD(2,2), &wsaData);
010
011 if(ret != 0)
012
013 {
014
015 MessageBox("初始化网络协议失败!");
016
017 return FALSE;
018
019 }
020
021 file://创建服务器端套接字
022
023 ServerSock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
024
025 if(ServerSock == INVALID_SOCKET)
026
027 {
028
029 MessageBox("创建套接字失败!");
030
031 closesocket(ServerSock);
032
033 WSACleanup();
034
035 return FALSE;
036
037 }
038
039 file://绑定到本地一个端口上
040
041 sockaddr_in localaddr;
042
043 localaddr.sin_family = AF_INET;
044
045 localaddr.sin_port = htons(8888); file://端口号不要与其他应用程序冲突
046
047 localaddr.sin_addr.s_addr = 0;
048
049 if(bind(ServerSock ,(struct sockaddr*)&localaddr,sizeof(sockaddr))
050
051 = = SOCKET_ERROR)
052
053 {
054
055 MessageBox("绑定地址失败!");
056
057 closesocket(ServerSock);
058
059 WSACleanup();
060
061 return FALSE;
062
063 }
064
065 file://将SeverSock设置为异步非阻塞模式,并为它注册各种网络异步事件,其 中 m_hWnd
066
067 file://为应用程序的主对话框或主窗口的句柄
068
069 if(WSAAsyncSelect(ServerSock, m_hWnd, NETWORK_EVENT,
070
071 FD_ACCEPT | FD_CLOSE | FD_READ | FD_WRITE) == SOCKET_ERROR)
072
073 {
074
075 MessageBox("注册网络异步事件失败!");
076
077 WSACleanup();
078
079 return FALSE;
080
081 }
082
083 listen(ServerSock, 5); file://设置侦听模式
084
085 return TRUE;
086
087 }
088
089 下面定义网络异步事件的回调函数
090
091 void CSocketSeverDlg::OnNetEvent(WPARAM wParam, LPARAM lParam)
092
093 {
094
095 file://调用Winsock API函数,得到网络事件类型
096
097 int iEvent = WSAGETSELECTEVENT(lParam);
098
099 file://调用Winsock API函数,得到发生此事件的客户端套接字
100
101 SOCKET CurSock= (SOCKET)wParam;
102
103 switch(iEvent)
104
105 {
106
107 case FD_ACCEPT: file://客户端连接请求事件
108
109 OnAccept(CurSock);
110
111 break;
112
113 case FD_CLOSE: file://客户端断开事件:
114
115 OnClose(CurSock);
116
117 break;
118
119 case FD_READ: file://网络数据包到达事件
120
121 OnReceive(CurSock);
122
123 break;
124
125 case FD_WRITE: file://发送网络数据事件
126
127 OnSend(CurSock);
128
129 break;
130
131 default: break;
132
133 }
134
135 }





以下是发生在相应Socket上的各种网络异步事件的处理函数,其中OnAccept传进来的参数是服务器端创建的套接字,OnClose()、OnReceive()和OnSend()传进来的参数均是服务器端在接受客户端连接时新创建的用与此客户端通信的Socket。
显示代码
打印
01 void CSocketSeverDlg::OnAccept(SOCKET CurSock)
02
03 {
04
05 file://接受连接请求,并保存与发起连接请求的客户端进行通信Socket
06
07 file://为新的socket注册异步事件,注意没有Accept事件
08
09 }
10
11 void CSocketSeverDlg::OnClose(SOCET CurSock)
12
13 {
14
15 file://结束与相应的客户端的通信,释放相应资源
16
17 }
18
19 void CSocketSeverDlg::OnSend(SOCET CurSock)
20
21 {
22
23 file://在给客户端发数据时做相关预处理
24
25 }
26
27 void CSocketSeverDlg::OnReceive(SOCET CurSock)
28
29 {
30
31 file://读出网络缓冲区中的数据包
32
33 }





用同样的方法建立一个客户端应用程序。初始化网络部分,不需要将套接字设置为监听模式。注册异步事件时,没有FD_ACCEPT,但增加了FD_CONNECT事件,因此没有OnAccept()函数,但增加了OnConnect()函数。向服务器发出连接请求时,使用connect()函数,连接成功后,会响应到OnConnect()函数中。下面是OnConnect()函数的定义,传进来的参数是客户端Socket和服务器端发回来的连接是否成功的标志。
显示代码
打印
01 void CSocketClntDlg::OnConnect(SOCKET CurSock, int error)
02
03 {
04
05 if(0 = = error)
06
07 {
08
09 if(CurSock = = ClntSock)
10
11 MessageBox("连接服务器成功!");
12
13 }
14
15 }





定义OnReceive()函数,处理网络数据到达事件;

定义OnSend()函数,处理发送网络数据事件;

定义OnClose()函数,处理服务器的关闭事件。

以上就是用基于Windows消息机制的异步I/O模型做服务器和客户端应用程序的基本方法。另外还可以用事件模型、重叠模型或完成端口模型,读者可以参考有关书籍。

在实现了上面的例子后,你将对Winsock编网络通信程序的机制有了一定的了解。接下来你可以进行更精彩的编程, 不仅可以在网上传输普通数据,而且还以传输语音、视频数据,你还可以自己做一个网络资源共享的服务器软件,和你的同学在实验室的局域网里可以共同分享你的成果。
分享到:
评论

相关推荐

    QTcpSocket通信编程时阻塞与非阻塞的问题 - findumars - 博客园1

    1. 编程理解(36) 1. 经典资料(15) 1. 开源项目(16) 9.SaaS-云存储,云服务(38) 9.SaaS-云计算-学习(38)

    【Socket编程】--TCP异步通讯一服务器多客户端

    该资源是基于tcp的异步通讯,不是多线程,实现了一服务器多客户端

    TCPSocket编程.zip

    Socket简介 Windows Socket Linux Socket Socket常用函数介绍 TCP/IP网络程序框架与实例 通信方式 阻塞 非阻塞

    linux-c++ socket编程

    linux socket编程server和多client连接c++代码;select非阻塞监听;可直接运行./tcp-server和./tcp-client测试;移植性好;

    TCPSocket编程

    Socket简介 Windows Socket Linux Socket Socket常用函数介绍 TCP/IP网络程序框架与实例 通信方式 阻塞 非阻塞

    非阻塞的Socket类20150331

    使用接口SocketIO简化Socket编程。一步建立Client,两步建立Server,一步打开UDP,全部非阻塞操作。

    linux 非堵塞tcp/server tcp/client

    linux非堵塞socket tcp编程

    socket网络编程服务端程序支持多客户端

    1.采用重叠I/O方式实现的socket网络编程,异步非阻塞方式,代码效率比阻塞式的socket编程方式高。2.实现了TCP server方式,只用于服务端,可以支持多客户端。3.可以使用在各种场合用于监控网络数据。4.代码封装成库...

    Python 中的 Socket 编程

    Python 中的 Socket 编程 说明 译者注 授权 开始 Socket API 概览 TCP Sockets 客户端 / 服务器echo 程序 echo 程序的服务端 echo 程序的客户端 运行echo 程序的客户端和服务端 查看 socket 状态 通信流程的分解 ...

    C++ boost::asio编程-同步TCP详解及实例代码

    boost::asio编程-同步TCP boost.asio库是一个跨平台的网络及底层IO的C++...在服务器端,我会做个socket交给acceptor对象,让它一直等客户端连进来,连上以后再通过这个socket与客户端通信, 而所有的通信都是以阻塞方

    MFC网络编程之自制浏览器

    网络编程,当然要用到Windows Socket(套接字)技术。Socket相关的操作由一系列...因此,建议初学者学习编网络通信程序时,暂且不要用MFC提供的类,而先用Winsock2 API,这样有助于对异步、非阻塞Socket编程机制的理解。

    Linux非阻塞套接字client

    linux系统下的非阻塞套接字编程的客户端实现

    阻塞与非阻塞

    最近帮一哥们做一个简单的通信演示小程序,重拾遗忘很久的Windows网络编程,通过此次演示程序的制作,对于TCP的三路握手、Socket的运用理解更加深入了,同时从文字上的阻塞与非阻塞到情真意切的感受到二者之间的差异...

    python实现单线程多任务非阻塞TCP服务端

    本文实例为大家分享了python实现单线程多任务非阻塞TCP服务端的具体代码,供大家参考,具体内容如下 # coding:utf-8 from socket import * # 1.创建服务器socket sock = socket(AF_INET, SOCK_STREAM) # 2.绑定主机...

    Linux网络编程超级详细笔记

    网络通信模型:Linux支持多种网络通信模型,包括阻塞式和非阻塞式通信、多路复用、信号驱动等。需要了解不同的通信模型的优缺点和使用方式。 网络编程库:Linux下有多种网络编程库,包括libnet、libpcap等。这些库...

    windows下设置socket的connect超时

    2.将该socket设置为非阻塞模式 3.调用connect() 4.使用select()检查该socket描述符是否可写(注意,是可写) 5.根据select()返回的结果判断connect()结果 6.将socket设置为阻塞模式(如果你的程序不需要用阻塞...

    vs2005写的TCP 异步通讯编程实例

    利用socket实现服务器与客户端异步通讯 非阻塞模式 对初学着很有帮助!

    java socket 大文件传输,快速传输(包的分片,组装)源码

    java socket 大文件传输,快速传输, 数据包的分片,组装,涉及UDP,TCP传输技术,NIO非阻塞等等,适合对socket编程进一步学习的同学

    TcpScan(Console&MFC)

    里面分为两个程序,一个是Console下面的,一个是MFC的,其中Console中socket是阻塞select模式,MFC下面的是WSAAsyncSelect异步非阻塞模式。 都是利用Tcp Connect扫描端口,都可以批量ip,批量端口扫描,都是多线程的...

    《对话框》之《SOCKET类的设计和实现》

    首先介绍WinSock API函数,讲解阻塞/非阻塞的概念;然后介绍socket的使用。 1. WinSock API Socket接口是网络编程(通常是TCP/IP协议,也可以是其他协议)的API。最早的Socket接口是Berkeley接口,在Unxi操作系统中...

Global site tag (gtag.js) - Google Analytics