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. 使用Django时需要注意的八个要点

    1.在settings.py中使用os. path.dirname() 常用代码如下: # settings.py import os PROJECT_DIR = os.path.dirname(__ ...

  2. nginx 默认配置语法和日志的format

    nginx 默认配置 查看有nginx哪些默认配置文件,打开/etc/nginx/nginx.conf文件,查看尾行部分 会默认将/etc/nginx/conf.d/文件下其他以.conf结尾的配置文 ...

  3. LNMP环境搭建之编译安装指南(php-5.3.27.tar.gz)

    测试环境:CentOS release 6.5 (Final) 软件安装:nginx   mysql-5.5.32-linux2.6-x86_64.tar.gz   php-5.3.27.tar.gz ...

  4. idou老师教你学istio30:Mixer Redis Quota Adapter 实现和机制

    1. 配置 1.1参数 1.2 Params.Quota 1.3Params.Override 1.4Params.QuotaAlgorithm 速率限制的算法: Fixed Window 算法每个时 ...

  5. js中的forEach和map的区别

    我们先来看两者之间的相同之处 var arr = ['a','b','c','d']; arr.forEach(function(item,index,arr){ //item表示数组中的每一项,in ...

  6. 如何写好demo——学习感悟

    文章标题:教你如何写好Demo应用 如何制作出最有用的demo呢? 简,易 在demo中,我们要专注于单一的主题.我们的教学覆盖了很大的知识范围,因此,化整为零是非常必要的. 例如,我们要说明Andr ...

  7. 集合(python)

    # -*- coding: utf-8 -*- class Array(object): def __init__(self, size=32, init=None): self._size = si ...

  8. SQL进阶17-变量的声明/使用(输出)--全局变量/会话变量--用户变量/局部变量

    /*进阶17 变量 系统变量: 全局变量: 会话变量: 自定义变量: 用户变量: 局部变量: */ /* #一: 系统变量 #说明: 变量由系统提供,不是用户定义的,属于服务器层面 #使用的语法 #1 ...

  9. SignalR 初体验

    目录 一.前言 二.服务端 2.1.站点服务端 2.2.宿主服务或客户端 2.3.持久连接和集线器 三.客户端 3.1.使用代理客户端 3.2.不使用代理客户端 一.前言 微软官方给的说明:ASP.N ...

  10. tslint.json的配置项说明

    tslint.json的配置项说明   extends: 内设配置项名称 rules: 规则 { //ts专用 adjacent-overload-signatures : true, // Enfo ...