TCP客户与服务器进程之间发生的重大事件时间表

TCP服务器

socket() --- bind() --- listen() --- accept() --- read() --- write --- read() --- close

TCP客户

socket() --- connect() --- write() --- read()  --- close()

套接字函数简介

int socket(int family, int type, int protocol);

指定要用的通信协议类型

int connect(int sockfd, const struct sockaddr *servaddr, socklen_t addrlen);

客户用connect()来建立与TCP服务器的连接

int bind(int sockfd, const struct sockaddr *myaddr, scoklen_t addrlen);

把一个本地协议地址赋予一个套接字

int listen(int sockfd, int backlog);

设置套接字为被动套接字,指示内核接收向该套接字的连接请求

int accept(int sockfd, struct sockaddr *cliaddr, socklen_t *addrlen);

由TCP服务器调用, 用来从已完成连接队列队头返回下一个已完成连接

pid_t fork(void);

产生子进程

int close(int sockfd);

关闭套接字, 并终止TCP连接

典型的并发服务器程序轮廓

  1. pid_t pid;
  2. int listenfd, connfd;
  3. listenfd=Socket(...);
  4. Bind(listenfd, ...);
  5. Listen(listenfd,LISTENQ);
  6. for(;;){
  7. connfd=Accept(listenfd, ...);
  8. if((pid=Fork())==0){//如果是子进程
  9. Close(listenfd);//子进程关闭其监听套接字
  10. doit(connfd);//处理需求
  11. Close(connfd);//关闭子进程的已连接套接字
  12. exit(0);//子进程终止
  13. }
  14. Close(connfd);//父进程关闭已连接套接字
  15. }

TCP echo服务器程序

  1. #include "unp.h"
  2. #include <time.h>
  3. void srv_echo(int sockfd)
  4. {
  5. ssize_t n;
  6. char buf[1000];
  7. again:
  8. while((n=read(sockfd,buf,1000))>0)
  9. Writen(sockfd,buf,n);
  10. if(n<0 && errno==EINTR)
  11. goto again;
  12. else if(n<0)
  13. err_sys("srv_echo: read error");
  14. }
  15. int
  16. main(int argc,char ** argv)
  17. {
  18. int listenfd, connfd;
  19. pid_t childpid;
  20. socklen_t clilen;
  21. struct sockaddr_in cliaddr, servaddr;
  22. char buff[1000];
  23. listenfd=Socket(AF_INET, SOCK_STREAM, 0);
  24. bzero(&servaddr, sizeof(servaddr));
  25. servaddr.sin_family=AF_INET;
  26. servaddr.sin_addr.s_addr=htonl(INADDR_ANY);
  27. servaddr.sin_port=htons(1234);
  28. Bind(listenfd, (SA*) &servaddr, sizeof(servaddr));
  29. Listen(listenfd, 100);
  30. for(;;){
  31. clilen=sizeof(cliaddr);
  32. connfd=Accept(listenfd, (SA*) &cliaddr, &clilen);
  33. printf("connection from %s , port %d\n",
  34. inet_ntop(AF_INET,&cliaddr.sin_addr, buff, sizeof(buff)),
  35. ntohs(cliaddr.sin_port));
  36. if((childpid=Fork())==0){
  37. Close(listenfd);
  38. srv_echo(connfd);
  39. exit(0);
  40. }
  41. Close(connfd);
  42. }
  43. }

TCP echo客户端程序

  1. #include "unp.h"
  2. void cli_echo(FILE *fp, int sockfd)
  3. {
  4. char sendline[1000], recvline[1000];
  5. while(Fgets(sendline,1000, fp)!=NULL){
  6. Writen(sockfd, sendline, strlen(sendline));
  7. if(Readline(sockfd, recvline, 1000)==0)
  8. err_quit("cli_echo: server terminated prematurely");
  9. Fputs(recvline, stdout);
  10. }
  11. }
  12. int
  13. main(int argc, char **argv)
  14. {
  15. int sockfd;
  16. struct sockaddr_in servaddr;
  17. if(argc!=2)
  18. err_quit("usage: tcpcli <IPaddress>");
  19. sockfd=Socket(AF_INET, SOCK_STREAM, 0);
  20. bzero(&servaddr, sizeof(servaddr));
  21. servaddr.sin_family=AF_INET;
  22. servaddr.sin_port=htons(1234);
  23. Inet_pton(AF_INET,argv[1], &servaddr.sin_addr);
  24. Connect(sockfd, (SA*) &servaddr, sizeof(servaddr));
  25. cli_echo(stdin,sockfd);
  26. exit(0);
  27. }

这一章的API讲解非常有逻辑性, 讲解的过程可以看到这样设计的合理性

4.5 listen函数

当一个客户的SYN到达, 若服务器的相应监听套接字维护的队列已经满了,  TCP就会忽略该分节, 也就是不发送RST. (这里的TCP指的是实现TCP的内核)

之所以这样设计有两点理由:

(1)这种情况是暂时的, TCP的正常重传机制会处理这个问题

(2)客户无法区分这个RST意思是 "该端口没有服务器在监听" 还是 "该端口有服务器在监听, 不过队列已满"

第四章 基本TCP套接字编程 第五章 TCP客户/服务器程序实例的更多相关文章

  1. 【UNIX网络编程(四)】TCP套接字编程具体分析

    引言: 套接字编程事实上跟进程间通信有一定的相似性,可能也正由于此.stevens这位大神才会将套接字编程与进程间的通信都归为"网络编程",并分别写成了两本书<UNP1> ...

  2. 基本套接字编程(1) -- tcp篇

    1. Socket简介 Socket是进程通讯的一种方式,即调用这个网络库的一些API函数实现分布在不同主机的相关进程之间的数据交换. 几个定义: (1)IP地址:即依照TCP/IP协议分配给本地主机 ...

  3. UNP学习笔记1——基本TCP套接字编程

    1 套接字地址结构 大多数套接字函数都需要一个指向套接字地址结构的指针作为参数.每个协议族都定义了自己的套接字结构.这些套接字的结构以sockaddr_开头,以每个协议族唯一的后缀名结尾. 1.1 I ...

  4. TCP套接字编程模型及实例

    摘要:     本文讲述了TCP套接字编程模块,包括服务器端的创建套接字.绑定.监听.接受.读/写.终止连接,客户端的创建套接字.连接.读/写.终止连接.先给出实例,进而结合代码分析. PS:本文权当 ...

  5. UNP学习笔记(第四章 基本TCP套接字编程)

    本章讲解编写一个完整的TCP客户/服务器程序所需要的基本套接字函数. socket函数 #include <sys/socket.h> int socket(int family,int ...

  6. unix网络编程第四章----基于TCP套接字编程

    为了执行网络I/O操作.进程必须做的第一件事情就是调用Socket函数.指定期待的通信协议 #include<sys/socket.h> int socket(int family,int ...

  7. UNP——第四章,TCP套接字编程

    1.socket 函数 首先被调用的函数,用于选择通信协议. socket调用成功后,得到的套接字为主动套接字CLOSED状态. PF 和 AF 的关系 PF的是协议族,AF是地址族,理论上一个PF包 ...

  8. 初探网络编程--TCP套接字编程演示

    今天看了一下<计算机网络:自顶向下方法>,也就是计算机网络的教材的应用层一章,决定实现以下后面的Java C/S应用程序的例子,用来演示TCP和UDP套接字编程. 程序流程如下: 1.一台 ...

  9. TCP套接字编程

    一.套接字(socket)函数 图1给出了在一个TCP客户与服务器通信的流程.服务器首先启动,稍后某个客户启动,它试图连接到服务器.假设客户给服务器发送一个请求,服务器处理该请求,并且给客户发回一个相 ...

随机推荐

  1. 在浏览器中输入www.taobao.com后执行的全部过程

    >>>点击网址后,应用层的DNS协议会将网址解析为IP地址: DNS查找过程: 1.        浏览器会检查缓存中有没有这个域名对应的解析过的IP地址,如果缓存中有,这个解析过程 ...

  2. mysql存储过程事务

    之前在写一个存储过程的时候由于不仔细导致数据库锁死,这里反省一下. 存储过程是这样的:把数据按顺序插入三张表,如果其中任何一处出错,就把前面已经做了的操作进行回滚,存储过程里面是用事务实现的,我是这么 ...

  3. kubeadm init初始化报错解决,亲测

    [preflight] You can also perform this action in beforehand using 'kubeadm config images pull' error ...

  4. 编译安装的httpd实现服务脚本,通过service和chkconfig进行管理

    把编译安装的httpd 实现服务脚本,通过service和chkconfig 进行管理 1 编译安装httpd 把httpd编译安装在/app/httpd/目录下. 2 在/etc/rc.d/init ...

  5. Linux下Mysql服务安装【1】

    https://www.cnblogs.com/xiaxiaoxu/p/9978976.html 第一步:获取mysql8.0的yum源 进入mysql官网获取RPM包下载地址 https://dev ...

  6. go常量的定义和枚举类型

    const a,b int = 1,2 const a,b     = 1,2 const ( a = "hello" b,c =3,4 ) 常量数值可作为各种类型使用 枚举类型的 ...

  7. MOS管做开关之初理解

    杰杰 物联网IoT开发 2017-10-12 大家好,我是杰杰.       今晚,设计电路搞了一晚上,终于从模电渣渣的我,把MOS管理解了那么一丢丢,如有写的不好的地方,请指出来.谢谢.      ...

  8. datafram 操作集锦

    Spark Python API 官方文档中文版> 之 pyspark.sql (二) 2017-11-04 22:13 by 牛仔裤的夏天, 365 阅读, 0 评论, 收藏, 编辑 摘要:在 ...

  9. Vue.js-组件化前端开发新思路

    Vue.js-组件化前端开发新思路 12017.04.14 18:31:25字数 6228阅读 5632 本文章是我最近在公司的一场内部分享的内容.我有个习惯就是每次分享都会先将要分享的内容写成文章. ...

  10. cf242 E

    题意: $n$ 个数 $a_i$, 两种询问 $1, l, r$ 查询 $[l, r]$ 的和 $2, l, r, x$ 将区间 $[l, r]$ 所有数异或 $x$ 建立 $30$ 课线段树 第 $ ...