网络编程-一个简单的echo程序(0)

青旅半醒 2021-12-11 11:37 378阅读 0赞

来源:公众号【编程珠玑】
作者:守望先生

前言

在上一篇《网络编程-从TCP连接的建立说起》中简单介绍了TCP连接的建立,本文暂时先抛开TCP更加详细的介绍,来看看如何实现一个简单的网络程序。

一个简单的echo程序

本文以及后续文章都将会围绕该程序进行介绍。程序大体流程如下:

640?wx\_fmt=png

echo程序

首先启动服务端,客户端通过TCP的三次握手与服务端建立连接;而后,客户端发送一段字符串,服务端收到字符串后,原封不动的发回给客户端。

我们先将代码呈现,后面再进行更加详细的解释。
客户端代码client.c如下:

  1. //client.c
  2. //来源:公众号【编程珠玑】网站:https://www.yanbinghu.com
  3. #include<stdio.h>
  4. #include<string.h>
  5. #include<stdlib.h>
  6. #include<unistd.h>
  7. #include <arpa/inet.h>
  8. #include<sys/socket.h>
  9. #include<sys/types.h>
  10. #define MAXLINE 128
  11. int main(int argc, char **argv)
  12. {
  13. int sockfd; //连接描述符
  14. struct sockaddr_in servaddr;//socket结构信息
  15. char sendMsg[MAXLINE] = {
  16. 0};
  17. char recvMsg[MAXLINE] = {
  18. 0};
  19. //检查参数数量
  20. if (argc < 2)
  21. {
  22. printf("usage: ./client ip port\n");
  23. return -1;
  24. }
  25. //初始化结构体
  26. bzero(&servaddr, sizeof(servaddr));
  27. //指定协议族
  28. servaddr.sin_family = AF_INET;
  29. //第一个参数为ip地址,需要把ip地址转换为sin_addr类型
  30. inet_pton(AF_INET, argv[1], &servaddr.sin_addr);
  31. //第二个参数为端口号
  32. servaddr.sin_port = htons(atoi(argv[2]));
  33. sockfd = socket(AF_INET, SOCK_STREAM, 0);
  34. if(-1 == sockfd)
  35. {
  36. perror("socket error");
  37. return -1;
  38. }
  39. //连接服务器,如果非0,则连接失败
  40. if(0 != connect(sockfd, (const struct sockaddr *)&servaddr, sizeof(servaddr)))
  41. {
  42. perror("connect failed");
  43. return -1;
  44. }
  45. //从控制台读取消息
  46. if(NULL !=fgets(sendMsg,MAXLINE,stdin))
  47. {
  48. write(sockfd, sendMsg, strlen(sendMsg));
  49. }
  50. if(0 != read(sockfd, recvMsg, MAXLINE))
  51. {
  52. printf("recv msg:%s\n",recvMsg);
  53. }
  54. close(sockfd);
  55. return 0;
  56. }

服务端代码server.c如下:

  1. //server.c
  2. //来源:公众号【编程珠玑】网站:https://www.yanbinghu.com
  3. #include<stdio.h>
  4. #include<string.h>
  5. #include<stdlib.h>
  6. #include<unistd.h>
  7. #include <arpa/inet.h>
  8. #include<sys/socket.h>
  9. #include<sys/types.h>
  10. #define SERV_PORT 1234
  11. #define MAXLINE 128
  12. int main(int argc, char **argv)
  13. {
  14. int listenfd = 0;//监听描述符
  15. int connfd = 0; //已连接描述符
  16. socklen_t clilen;
  17. char recvMsg[MAXLINE] = {
  18. 0};
  19. //服务器和客户端socket信息
  20. struct sockaddr_in cliaddr, servaddr;
  21. char ip[MAXLINE] = {
  22. 0};
  23. //初始化服务端socket信息
  24. bzero(&servaddr, sizeof(servaddr));
  25. servaddr.sin_family = AF_INET;
  26. //如果输入ip和端口,使用输入的ip和端口
  27. if(3 == argc)
  28. {
  29. inet_pton(AF_INET, argv[1], &servaddr.sin_addr);
  30. servaddr.sin_port = htons(atoi(argv[2]));
  31. }
  32. else
  33. {
  34. //使用默认的ip和port
  35. servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
  36. servaddr.sin_port = htons(SERV_PORT);
  37. }
  38. listenfd = socket(AF_INET, SOCK_STREAM, 0);
  39. if(-1 == listenfd)
  40. {
  41. perror("socket error");
  42. return -1;
  43. }
  44. //绑定指定ip和端口
  45. if(0 != bind(listenfd, (struct sockaddr *) &servaddr, sizeof(servaddr)))
  46. {
  47. perror("bind error");
  48. return -1;
  49. }
  50. printf("start server at %s:%d\n",inet_ntop(AF_INET,&servaddr.sin_addr,ip,MAXLINE),ntohs(servaddr.sin_port));
  51. listen(listenfd, 4);
  52. //处理来自客户端的连接
  53. clilen = sizeof(cliaddr);
  54. connfd = accept(listenfd, (struct sockaddr *)&cliaddr, &clilen);
  55. if(-1 == connfd)
  56. {
  57. perror("accept failed");
  58. return -1;
  59. }
  60. printf("connect from %s %d\n",inet_ntop(AF_INET,&cliaddr.sin_addr,ip,MAXLINE),ntohs(cliaddr.sin_port));
  61. //读取客户端发送的消息
  62. if(0 != read(connfd, recvMsg, MAXLINE))
  63. {
  64. printf("recv msg:%s\n",recvMsg);
  65. }
  66. //将读取内容原封不动地发送回去
  67. write(connfd, recvMsg, MAXLINE);
  68. close(connfd);
  69. close(listenfd);
  70. return 0;
  71. }

编译运行

编译客户端服务端代码:

  1. $ gcc -o client client.c
  2. $ gcc -o server server.c

在两个终端分别运行server和client。

  1. $ ./server
  2. start server at 0.0.0.0:1234start server at 0.0.0.0:1234

运行客户端,并输入内容:

  1. $ ./client 127.0.0.1 1234
  2. hello 编程珠玑

服务端最终打印:

  1. start server at 0.0.0.0:1234
  2. connect from 127.0.0.1 47536
  3. recv msg:hello 编程珠玑

客户端最终打印:

  1. hello 编程珠玑
  2. recv msg:hello 编程珠玑

从运行结果可以看到,客户端连接到服务端后,发送一段字符串“hello 编程珠玑”后,服务端返回同样的字符串,达到了我们想要的目的。当然代码里有很多地方还需要完善,但这不影响我们对网络编程的学习。

整体流程说明

整体流程可结合下图来理解:

640?wx\_fmt=png

  1. TCP的三次握手,我们在《网络编程-从TCP连接的建立说起》中就已经介绍了。在图中,标示了在调用某些接口后的状态。例如,服务端在调用socketbindlisten等函数后,处于LISTEN状态;客户端调用connect函数并返回后,完成三次握手,客户端与服务端都处于ESTABLISHED状态。

这些状态我们是可以观察到的,首先在一个终端启动服务器:

  1. $ ./server
  2. start server at 0.0.0.0:1234

在另外一个终端使用netstat命令(或使用ss命令)观察:

  1. $ netstat -anp |grep :1234
  2. tcp 0 0 0.0.0.0:1234 0.0.0.0:* LISTEN 17730/server

netstat命令的使用可参考netstat命令详解,可以看到server程序当前处于LISTEN状态。

而如果客户端进行连接后再观察会发现:

  1. $ netstat -anp |grep :1234
  2. tcp 0 0 0.0.0.0:1234 0.0.0.0:* LISTEN 17730/server
  3. tcp 0 0 127.0.0.1:48094 127.0.0.1:1234 ESTABLISHED 17957/client
  4. tcp 0 0 127.0.0.1:1234 127.0.0.1:48094 ESTABLISHED 17730/server

从结果中看到,客户端此时处于ESTABLISHED状态,而服务端有一条连接处于ESTABLISHED状态,还有一条处于LISTEN状态,这是为何呢?我们后面再解释。
由于三次握手的过程非常快,其他的状态我们不是很方便能观察到。

那么结合代码,整个流程又是怎样的呢?请看下图:

640?wx\_fmt=png

客户端-服务端

在弄清楚图中的接口含义之前,实际上你可以认为客户端连接服务器的整个过程你可以看成是这样的(阻塞模式):

当然了,我们需要注意到的是:

对于图中所提到的接口和数据结构的介绍和使用说明都会在后面进行详细介绍。

小结

看到这里,想必你对我们的echo程序的整体已经有了大致的了解。在对这些接口和数据结构进行详细介绍之前,你可以将代码复制并进行编译运行,观察文中提到的内容,下一节将进行更加详细的介绍。

本文也可通过阅读原文或以下地址访问:
https://www.yanbing.com/2019/07/07/40135.html

问题

关注公众号【编程珠玑】,获取更多Linux/C/C++/Python/Go/算法/工具等原创技术文章。后台免费获取经典电子书和视频资源

640?wx\_fmt=jpeg

发表评论

表情:
评论列表 (有 0 条评论,378人围观)

还没有评论,来说两句吧...

相关阅读