VB.net 2010 视频教程 VB.net 2010 视频教程 python基础视频教程
SQL Server 2008 视频教程 c#入门经典教程 Visual Basic从门到精通视频教程
当前位置:
首页 > 网络工程 > 网络工程师 >
  • 如何让服务器端持续不断地监听客户端的请求?

前面的程序,不管服务器端还是客户端,都有一个问题,就是处理完一个请求立即退出了,没有太大的实际意义。能不能像Web服务器那样一直接受客户端的请求呢?能,使用 while 循环即可。

修改前面的回声程序,使服务器端可以不断响应客户端的请求。

服务器端 server.cpp:

 
  1. #include <stdio.h>
  2. #include <winsock2.h>
  3. #pragma comment (lib, "ws2_32.lib") //加载 ws2_32.dll
  4.  
  5. #define BUF_SIZE 100
  6.  
  7. int main(){
  8. WSADATA wsaData;
  9. WSAStartup( MAKEWORD(2, 2), &wsaData);
  10.  
  11. //创建套接字
  12. SOCKET servSock = socket(AF_INET, SOCK_STREAM, 0);
  13.  
  14. //绑定套接字
  15. sockaddr_in sockAddr;
  16. memset(&sockAddr, 0, sizeof(sockAddr)); //每个字节都用0填充
  17. sockAddr.sin_family = PF_INET; //使用IPv4地址
  18. sockAddr.sin_addr.s_addr = inet_addr("127.0.0.1"); //具体的IP地址
  19. sockAddr.sin_port = htons(1234); //端口
  20. bind(servSock, (SOCKADDR*)&sockAddr, sizeof(SOCKADDR));
  21.  
  22. //进入监听状态
  23. listen(servSock, 20);
  24.  
  25. //接收客户端请求
  26. SOCKADDR clntAddr;
  27. int nSize = sizeof(SOCKADDR);
  28. char buffer[BUF_SIZE] = {0}; //缓冲区
  29. while(1){
  30. SOCKET clntSock = accept(servSock, (SOCKADDR*)&clntAddr, &nSize);
  31. int strLen = recv(clntSock, buffer, BUF_SIZE, 0); //接收客户端发来的数据
  32. send(clntSock, buffer, strLen, 0); //将数据原样返回
  33.  
  34. closesocket(clntSock); //关闭套接字
  35. memset(buffer, 0, BUF_SIZE); //重置缓冲区
  36. }
  37.  
  38. //关闭套接字
  39. closesocket(servSock);
  40.  
  41. //终止 DLL 的使用
  42. WSACleanup();
  43.  
  44. return 0;
  45. }


客户端 client.cpp:

 
  1. #include <stdio.h>
  2. #include <WinSock2.h>
  3. #include <windows.h>
  4. #pragma comment(lib, "ws2_32.lib") //加载 ws2_32.dll
  5.  
  6. #define BUF_SIZE 100
  7.  
  8. int main(){
  9. //初始化DLL
  10. WSADATA wsaData;
  11. WSAStartup(MAKEWORD(2, 2), &wsaData);
  12.  
  13. //向服务器发起请求
  14. sockaddr_in sockAddr;
  15. memset(&sockAddr, 0, sizeof(sockAddr)); //每个字节都用0填充
  16. sockAddr.sin_family = PF_INET;
  17. sockAddr.sin_addr.s_addr = inet_addr("127.0.0.1");
  18. sockAddr.sin_port = htons(1234);
  19.  
  20. char bufSend[BUF_SIZE] = {0};
  21. char bufRecv[BUF_SIZE] = {0};
  22.  
  23. while(1){
  24. //创建套接字
  25. SOCKET sock = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP);
  26. connect(sock, (SOCKADDR*)&sockAddr, sizeof(SOCKADDR));
  27. //获取用户输入的字符串并发送给服务器
  28. printf("Input a string: ");
  29. gets(bufSend);
  30. send(sock, bufSend, strlen(bufSend), 0);
  31. //接收服务器传回的数据
  32. recv(sock, bufRecv, BUF_SIZE, 0);
  33. //输出接收到的数据
  34. printf("Message form server: %s\n", bufRecv);
  35.  
  36. memset(bufSend, 0, BUF_SIZE); //重置缓冲区
  37. memset(bufRecv, 0, BUF_SIZE); //重置缓冲区
  38. closesocket(sock); //关闭套接字
  39. }
  40.  
  41. WSACleanup(); //终止使用 DLL
  42. return 0;
  43. }

先运行服务器端,再运行客户端,结果如下:
Input a string: c language
Message form server: c language
Input a string: C语言中文网
Message form server: C语言中文网
Input a string: 学习C/C++编程的好网站
Message form server: 学习C/C++编程的好网站

while(1) 让代码进入死循环,除非用户关闭程序,否则服务器端会一直监听客户端的请求。客户端也是一样,会不断向服务器发起连接。

需要注意的是:server.cpp 中调用 closesocket() 不仅会关闭服务器端的 socket,还会通知客户端连接已断开,客户端也会清理 socket 相关资源,所以 client.cpp 中需要将 socket() 放在 while 循环内部,因为每次请求完毕都会清理 socket,下次发起请求时需要重新创建。后续我们会进行详细讲解。

参考文章:http://c.biancheng.net/view/2348.html


相关教程