{ 在网络技术日益发展的今天,如果自己也能够编写一个实用的网络应用程序,那么,不仅能够激发对网络的兴趣,促使自己对网络知识的追求,同时开发过程本身也是一个很好的学习过程。 在VC6.0中MFC对网络编程有着很好的支持,针对不同用途的网络应用程序,VC有不同的封装类进行支持,如FTP、HTTP等等,使用户能够很快的开发出相应的程序,但同时,也使用户失去了一些了解网络程序运行的底层机制的机会,更重要的是也失去了一定程度上的灵活性。在此我们介绍利用SOCKET套接字进行开发的一般步骤,供读者参考。 在Windows和UNIX下编写网络应用程序,基本上都是利用SOCKET套接字进行数据通讯,SOCKET套接字是从UNIX环境下承袭而来,它在程序中的作用可以理解为网络数据通讯的一个代理,其在Windows中的设计思路与UNIX下相比基本没有多大变化,分为服务器套接字和客户机套接字两个设计部分,设计思路如下: 第一部分 服务器端 一、创建服务器套接字(CREATE)。 二、服务器套接字进行信息绑定(BIND),并开始监听连接(LISTEN)。 三、接受来自客户端的连接请求(ACCEPT),并创建接收进程。 四、开始数据传输(SEND、RECEIVE)。 五、关闭套接字(CLOSESOCKET)。 第二部分 客户机端 一、创建客户机套接字(CREATE)。 二、与远程服务器进行连接(CONNECT),如被接受则创建接收进程。 三、开始数据传输(SEND、RECEIVE)。 四、关闭套接字(CLOSESOCKET)。 以上的设计思路是我们开发的基本步骤,同时也是大多数网络应用程序运行的基本方式,下面我们具体说明它在VC中的实现。 服务器端: 一、建立支持SOCKET项目。 利用APP WIZARD创建MFC EXE项目,进行到WIZARD的第四步时,在“What features would you like include?”中,选中“Windows Sockets”项。其它各步骤各选项根据实际应用进行选择即可。这样创建的项目就已经支持SOCKET,并已经初始化了。 如果要在已有的项目中添加SOCKET支持,只须进行两项工作: 1在stdafx.h文件中包含头文件WINSOCK.H (#include “winsock.h” )。 2初始化套接字,在应用程序类的成员函数:“::InitInstance()”中添加如下初始化套接字代码。 if (!AfxSocketInit()) {AfxMessageBox(IDP_SOCKETS_INIT_FAILED); return FALSE;} 二、创建服务套接字并创建监听线程。 //创建服务套接字 SOCKET sercon=socket(PF_INET,SOCK_STREAM,0); //判断是否成功创建 if (sercon==INVALID_SOCKET) {AfxMessageBox(“Server WRONG !”); return -1;} //配置套接字地址等信息 SOCKADDR_IN sin; sin.sin_family=AF_INET; //指定本地地址 sin.sin_addr.s_addr=htonl(INADDR_ANY); //指定服务器端口号nPort,可自设 int nPort=5080; sin.sin_port=htons(nPort); //地址信息与套接字进行绑定。 if (bind(sercon,(LPSOCKADDR)&sin,sizeof(sin))==SOCKET_ERROR) {AfxMessageBox(“bind wrong!”); return -1;} //建立监听队列(大小为3),开始监听 if (listen(sercon,3)==SOCKET_ERROR) {AfxMessageBox(“listen wrong!”); return -1;}; ①实现监听线程,并创建数据接收线程。 //在程序需要开始监听连接的地方创建监听线程,并实现。 //创建监听线程(在程序开始或按钮事件实现中) AfxBeginThread(waitconnect,NULL); //实现监听线程 UINT waitconnect(LPVOID lpParm) {SOCKET conn[3]; int lenc=sizeof(sockaddr); int alreadycon=0; //sercon为前面所创建服务器套接字 while(1) {if (alreadycon<=3) {//接受连接请求 conn[alreadycon]=accept(sercon,&cin,&lenc); if (conn[alreadycon]==INVALID_SOCKET) {AfxMessageBox(“accept WRONG !”);} else {//创建数据接收线程 AfxBeginThread(readdata,&connn[alreadycon]); Alreadycon= alreadycon+1; return 0;}} else {//避免影响主线程运行 Sleep(200);} } } ②实现数据接收线程。 UINT waitconnect(LPVOID ss) { SOCKET *readsock; readsock=(SOCKET *)ss; char buf[2000]; int revnum=0; //开始循环接受数据 while (revnum!=-1) {//revnum<=0则表示连接已断! revnum=recv(*readsock,buf,2000,0); if (revnum>0) buf[revnum]=0;//截断缓冲区 //buf中存储已接受数据。} } ③发送数据 //conn[1]为用于接受连接的套接字, sendstr为所发送数据。 send(conn[1],LPCTSTR(sendstr),sendstr.GetLength(),0); ④关闭套接字。 //conn[1]为用于接受连接的套接字 closesocket(conn[1]); 客户程序端: 客户端程序的编程有很多与服务器端相同或相近,甚至相同的代码。 一、建立支持SOCKET项目。 方法同服务器端。 二、创建客户套接字、对服务器进行连接。 //nHost 须用户指定的远程服务机,IP或域名。 CString nHost; //h为地址信息 struct hostent *h; h=gethostbyname(nHost); //nHost 须用户指定的远程服务端口号 int nPort; SOCKET con_client; SOCKADDR_IN csin; if (h!=NULL) {//创建套接字 con_client =socket(AF_INET,SOCK_STREAM,0); csin.sin_family=AF_INET; memcpy(&(csin.sin_addr.s_addr),h->h_addr,sizeof(int)); csin.sin_port=htons(nPort); //开始连接 if (connect(con_client,(LPSOCKADDR)&csin,sizeof(csin))) {//AfxMessageBox(“connect wrong!”); return -1;} else {//连接成功,创建数据接收线程 AfxBeginThread(readdata,&con_client);} } 三、实现数据接收线程。 代码与服务器端完全相同。 四、发送数据。 //con_client 为与服务器进行连接的套接字。 send(con_client,LPCTSTR(sendstr),sendstr.GetLength(),0); 五、关闭套接字。 // con_client 为与服务器进行连接的套接字。 closesocket(conn[1]); 在实际应用中,应当根据需要调整并改变一些变量的作用域。 以上程序在VC6.0 、WIN NT4.0 及Windows 98中调试通过。 最后说一点,在VC6.0 MFC中的CSOCKET类是对SOCKET的一个MFC封装,并且它支持文档序列化,可以方便地实现不同数据类型的传输。本文前面之所以没有介绍CSOCKET,是因为用CSOCKET的实现方法与上面所讲述的思路相同,并且更为简单。另外一个更重要的原因是便于向UNIX编程时移植,因为UNIX支持SOCKET。} |