在VC6.0中编写网络应用程序

    技术2022-05-11  96

    {在VC6.0中编写网络应用程序} 》
    {  在网络技术日益发展的今天,如果自己也能够编写一个实用的网络应用程序,那么,不仅能够激发对网络的兴趣,促使自己对网络知识的追求,同时开发过程本身也是一个很好的学习过程。  在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。}
     

    最新回复(0)