1、ws2_32.lib)intmain()WSADATAwsaData;SOCKETsClient;SOCKADDR_INserver;charszMessageMSGSIZE;ret;/InitializesocketlibraryWSAStartup(0x0202,&wsaData);CreateclientsocketsClient=socket(AF_INET,SOCK_STREAM,IPPROTO_TCP);Connecttoservermemset(&server,0,sizeof(SOCKADDR_IN);server.sin_familyAF_INET;server.sin_ad
2、dr.S_un.S_addrinet_addr(SERVER_ADDRESS);server.sin_porthtons(PORT);connect(sClient,(structsockaddr*)&while(TRUE)printf(Send:);gets(szMessage);Sendmessagesend(sClient,szMessage,strlen(szMessage),0);Receiveretrecv(sClient,MSGSIZE,szMessageret0;Received%dbytes:%sn,ret,szMessage);Cleanupclosesocket(sCli
3、ent);WSACleanup();return0;客户端所做的事情相当简单,创建套接字,连接服务器,然后不停的发送和接收数据。比较容易想到的一种服务器模型就是采用一个主线程,负责监听客户端的连接请求,当接收到某个客户端的连接请求后,创建一个专门用于和该客户端通信的套接字和一个辅助线程。以后该客户端和服务器的交互都在这个辅助线程内完成。这种方法比较直观,程序非常简单而且可移植性好,但是不能利用平台相关的特性。例如,如果连接数增多的时候(成千上万的连接),那么线程数成倍增长,操作系统忙于频繁的线程间切换,而且大部分线程在其生命周期内都是处于非活动状态的,这大大浪费了系统的资源。所以,如果你已经知
4、道你的代码只会运行在Windows平台上,建议采用WinsockI/O模型。一.选择模型Select(选择)模型是Winsock中最常见的I/O模型。之所以称其为“Select模型”,是由于它的“中心思想”便是利用select函数,实现对I/O的管理。最初设计该模型时,主要面向的是某些使用UNIX操作系统的计算机,它们采用的是Berkeley套接字方案。Select模型已集成到Winsock1.1中,它使那些想避免在套接字调用过程中被无辜“锁定”的应用程序,采取一种有序的方式,同时进行对多个套接字的管理。由于Winsock1.1向后兼容于Berkeley套接字实施方案,所以假如有一个Berke
5、ley套接字应用使用了select函数,那么从理论角度讲,毋需对其进行任何修改,便可正常运行。(节选自Windows网络编程第八章)下面的这段程序就是利用选择模型实现的Echo服务器的代码(已经不能再精简了):winsock.hg_iTotalConng_CliSocketArrFD_SETSIZE;DWORDWINAPIWorkerThread(LPVOIDlpParameter);sListen,local,client;iaddrSizesizeof(SOCKADDR_IN);dwThreadId;listeningsListenBindlocal.sin_addr.S_un.S_add
6、rhtonl(INADDR_ANY);local.sin_familylocal.sin_portbind(sListen,Listenlisten(sListen,3);workerthreadCreateThread(NULL,WorkerThread,NULL,dwThreadId);Acceptaconnectionaccept(sListen,client,iaddrSize);Acceptedclient:%s:%dninet_ntoa(client.sin_addr),ntohs(client.sin_port);Addg_CliSocketArrg_CliSocketArrg_
7、iTotalConn+lpParam)i;fd_setfdread;structtimevaltv1,0;FD_ZERO(&fdread);for(iig_iTotalConn;i+)FD_SET(g_CliSocketArr,Weonlycarereadeventselect(0,fdread,tv);if(ret=0)Timeexpiredcontinue;(FD_ISSET(g_CliSocketArr,fdread)Aeventhappenedonrecv(g_CliSocketArr,0|SOCKET_ERRORWSAGetLastError()WSAECONNRESET)Clien
8、tclosed%dclosed.ng_CliSocketArr);closesocket(g_CliSocketArr);-1)g_CliSocketArri-g_CliSocketArr-g_iTotalConn;elsereceivedmessagefromclientsend(g_CliSocketArr,服务器的几个主要动作如下:1.创建监听套接字,绑定,监听;2.创建工作者线程;3.创建一个套接字数组,用来存放当前所有活动的客户端套接字,每accept一个连接就更新一次数组;4.接受客户端的连接。这里有一点需要注意的,就是我没有重新定义FD_SETSIZE宏,所以服务器最多支持的并发
9、连接数为64。而且,这里决不能无条件的accept,服务器应该根据当前的连接数来决定是否接受来自某个客户端的连接。一种比较好的实现方案就是采用WSAAccept函数,而且让WSAAccept回调自己实现的ConditionFunction。如下所示:CALLBACKConditionFunc(LPWSABUFlpCallerId,LPWSABUFlpCallerData,LPQOSlpSQOS,LPQOSlpGQOS,LPWSABUFlpCalleeId,LPWSABUFlpCalleeData,GROUPFAR*g,DWORDdwCallbackData)(当前连接数FD_SETSIZE)C
10、F_ACCEPT;CF_REJECT;工作者线程里面是一个死循环,一次循环完成的动作是:1.将当前所有的客户端套接字加入到读集fdread中;2.调用select函数;3.查看某个套接字是否仍然处于读集中,如果是,则接收数据。如果接收的数据长度为0,或者发生WSAECONNRESET错误,则表示客户端套接字主动关闭,这时需要将服务器中对应的套接字所绑定的资源释放掉,然后调整我们的套接字数组(将数组中最后一个套接字挪到当前的位置上)除了需要有条件接受客户端的连接外,还需要在连接数为0的情形下做特殊处理,因为如果读集中没有任何套接字,select函数会立刻返回,这将导致工作者线程成为一个毫无停顿的
11、死循环,CPU的占用率马上达到100%。关系到套接字列表的操作都需要使用循环,在轮询的时候,需要遍历一次,再新的一轮开始时,将列表加入队列又需要遍历一次.也就是说,Select在工作一次时,需要至少遍历2次列表,这是它效率较低的原因之一.在大规模的网络连接方面,还是推荐使用IOCP或EPOLL模型.但是Select模型可以使用在诸如对战类游戏上,比如类似星际这种,因为它小巧易于实现,而且对战类游戏的网络连接量并不大.对于Select模型想要突破Windows64个限制的话,可以采取分段轮询,一次轮询64个.例如套接字列表为128个,在第一次轮询时,将前64个放入队列中用Select进行状态查询
12、,待本次操作全部结束后.将后64个再加入轮询队列中进行轮询处理.这样处理需要在非阻塞式下工作.以此类推,Select也能支持无限多个.二.异步选择Winsock提供了一个有用的异步I/O模型。利用这个模型,应用程序可在一个套接字上,接收以Windows消息为基础的网络事件通知。具体的做法是在建好一个套接字后,调用WSAAsyncSelect函数。该模型最早出现于Winsock的1.1版本中,用于帮助应用程序开发者面向一些早期的16位Windows平台(如WindowsWorkgroups),适应其“落后”的多任务消息环境。应用程序仍可从这种模型中得到好处,特别是它们用一个标准的Windows例
13、程(常称为WndProc),对窗口消息进行管理的时候。该模型亦得到了MicrosoftFoundationClass(微软基本类,MFC)对象CSocket的采纳。我还是先贴出代码,然后做详细解释:tchar.hWM_SOCKETWM_USER+0LRESULTWndProc(HWND,UINT,WPARAM,LPARAM);WinMain(HINSTANCEhInstance,HINSTANCEhPrevInstance,PSTRszCmdLine,iCmdShow)staticTCHARszAppName_T(AsyncSelectModelHWNDhwndMSGmsgWNDCLASSwn
14、dclasswndclass.styleCS_HREDRAW|CS_VREDRAWwndclass.lpfnWndProcWndProcwndclass.cbClsExtrawndclass.cbWndExtrawndclass.hInstancehInstancewndclass.hIconLoadIcon(NULL,IDI_APPLICATION)wndclass.hCursorLoadCursorIDC_ARROW)wndclass.hbrBackground(HBRUSH)GetStockObject(WHITE_BRUSH)wndclass.lpszMenuNameNULLwndcl
15、ass.lpszClassNameszAppName(!RegisterClass(&wndclass)MessageBoxTEXT(ThisprogramrequiresNT!),szAppName,MB_ICONERROR)CreateWindow(szAppName,windowclassnamecaptionWS_OVERLAPPEDWINDOW,styleCW_USEDEFAULT,initialxpositionysizeparenthandlemenuinstanceNULL)creationparametersShowWindow(hwnd,iCmdShow);UpdateWi
16、ndow(hwnd);(GetMessage(&msg,0)TranslateMessage(&msg)DispatchMessage(&msg.wParam;(HWNDhwnd,UINTmessage,WPARAMwParam,LPARAMlParam)wsd;sListen;iAddrSizesizeof(client);switch(message)caseWM_CREATE:Socketwsd);sizeof(local);AssociatewithFD_ACCEPTWSAAsyncSelect(sListen,WM_SOCKET,FD_ACCEPT);WM_DESTROY:closesocket(sListen);PostQuitMessage(0);WM_SOCKET:(WSAGETSELECTERROR(lParam)closesocket(wParam);break;(WSAGETSELECTEVENT(lParam)FD_ACCEPT:connectionaccept(wParam,iAddrSize);FD_READandFD_CLOSEWSAAsyncSelect(sClient,FD_RE
copyright@ 2008-2022 冰豆网网站版权所有
经营许可证编号:鄂ICP备2022015515号-1