chapter6

  •  6.1 概述
  • I/O复用典型使用在下列网络应用场合。
    •   (1):当客户处理多个描述符时,必须使用IO复用
    •   (2):一个客户同时处理多个套接字是可能的,不过不叫少见。
    •   (3):如果一个TCP服务器既要处理监听套接字,又要处理已连接套接字。
    •   (4):如果一个服务器既要处理TCP,又要处理UDP
    •   (5):如果一个服务器要处理多个服务或多个协议
    •    IO复用并非只限于网络,许多重要的应用程序也需要使用这项技术。
  • 6.2 I/O模型
  • 在Unix下可用的5种I/O模型的基本区别:
    •   (1)阻塞式I/O
    •   (2)非阻塞式I/O
    •   (3)I/O复用(select和poll)
    •   (4)信号驱动式I/O(SIGIO)
    •   (5)异步I/O(POSIX的aio_系列函数)
    • 6.2.1 阻塞式I/O
    • 6.2.2 非阻塞式I/O模型
    • 6.2.3 I/O复用模型
      • 有个I/O复用,我们就可以调用select或poll,阻塞在这两个系统调用中的某一个之上,而不是阻塞在真正的I/O系统调用上。
    • 6.2.4 信号驱动式I/O模型
      • 我们也可以用信号,让内核在描述符就绪时发送SIGIO信号通知我们。我们称这种模型为信号驱动式I/O
    • 6.2.5 异步I/O模型
  •  6.3 select函数
    • 该函数允许进程指示内核等待多个事件中的任何一个发生,并且在有一个或多个事件发生或经历一段指定的时间后才唤醒它。
    • 我们调用select告知内核对那些描述符(就读写或异常)感兴趣以及等待多长时间。
    • #include <sys/select.h>
    • #include <sys/time.h>
    • int select(int maxfdp1, fd_set *readset, fd_set *writeset, fd_set *exceptset, cosnt struct timeval *timeout);
      • maxfdp1:最大的描述符ID+1
    • struct timeval
    • {
      • long tv_sec; //秒
      • long tv_usec;//微妙
    • };
    • void FD_ZERO(fd_set *fdset);
    • void FD_SET(int fd, fd_set *fdset);
    • void FD_CLR(int fd, fd_set *fdset);
    • int FD_ISSET(int fd, fd_set *fdset);
    • 6.3.1描述符就绪条件
      • (1):满足下列4个条件中的任何一个时,一个套接字准备好读

        • a):
        • ......
    • 6.3.2 select的最大描述符
  • 6.4 str_cli函数
  • 6.5 批量输入
  • 6.6 shutdown函数
    • 终止网络连接的方法不是调用close函数。不过close有两个限制,却可以使用shutdown来避免
    • (1):close把描述符的引用计数减1,但在该计数为0时才关闭套接字
    • (2):close终止读和写两个方向的数据传送。
      • #include <sys/socket.h>
      • int shutdown(int sockfd, int howto);
        • OK : 0 FAILED:-1
  • 6.8 TCP Service
    •  /* include fig01 */
      #include "unp.h" int
      main(int argc, char **argv)
      {
      int i, maxi, maxfd, listenfd, connfd, sockfd;
      int nready, client[FD_SETSIZE];
      ssize_t n;
      fd_set rset, allset;
      char buf[MAXLINE];
      socklen_t clilen;
      struct sockaddr_in cliaddr, servaddr; listenfd = Socket(AF_INET, SOCK_STREAM, ); bzero(&servaddr, sizeof(servaddr));
      servaddr.sin_family = AF_INET;
      servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
      servaddr.sin_port = htons(SERV_PORT); Bind(listenfd, (SA *) &servaddr, sizeof(servaddr)); Listen(listenfd, LISTENQ); maxfd = listenfd; /* initialize */
      maxi = -; /* index into client[] array */
      for (i = ; i < FD_SETSIZE; i++)
      client[i] = -; /* -1 indicates available entry */
      FD_ZERO(&allset);
      FD_SET(listenfd, &allset);
      /* end fig01 */ /* include fig02 */
      for ( ; ; ) {
      rset = allset; /* structure assignment */
      nready = Select(maxfd+, &rset, NULL, NULL, NULL); if (FD_ISSET(listenfd, &rset)) { /* new client connection */
      clilen = sizeof(cliaddr);
      connfd = Accept(listenfd, (SA *) &cliaddr, &clilen);
      #ifdef NOTDEF
      printf("new client: %s, port %d\n",
      Inet_ntop(AF_INET, &cliaddr.sin_addr, , NULL),
      ntohs(cliaddr.sin_port));
      #endif for (i = ; i < FD_SETSIZE; i++)
      if (client[i] < ) {
      client[i] = connfd; /* save descriptor */
      break;
      }
      if (i == FD_SETSIZE)
      err_quit("too many clients"); FD_SET(connfd, &allset); /* add new descriptor to set */
      if (connfd > maxfd)
      maxfd = connfd; /* for select */
      if (i > maxi)
      maxi = i; /* max index in client[] array */ if (--nready <= )
      continue; /* no more readable descriptors */
      } for (i = ; i <= maxi; i++) { /* check all clients for data */
      if ( (sockfd = client[i]) < )
      continue;
      if (FD_ISSET(sockfd, &rset)) {
      if ( (n = Read(sockfd, buf, MAXLINE)) == ) {
      /*4connection closed by client */
      Close(sockfd);
      FD_CLR(sockfd, &allset);
      client[i] = -;
      } else
      Writen(sockfd, buf, n); if (--nready <= )
      break; /* no more readable descriptors */
      }
      }
      }
      }
      /* end fig02 */
  • 6.9 pselect函数
    • #include <sys/select.h>
    • #include <signal.h>
    • #include <time.h>
    • int pselect(int masfdp1, fd_set *readset, fd_set *writeset, fd_set *exceptset,const struct timespec *timeout,const sigset_t *sigmask);
    • pselect相对于通常的select有两个变化
      • pselect使用timespec结构,而不使用timeval结构。

        • struct timespec{

          • time_t tv_sec;
          • long tv_nsec; // 纳秒书
        • };
      • pselect函数增加了第六个参数:一个指向信号掩码的指针。
        • ......
  • 6.10 poll函数
    • #include <poll.h>
    • int poll(struct pollfd *fdarray,unsigned long nfds, int timeout);
    • 如果有就绪描述符则返回去数目,若超时则为0,若出错则为-1
    • struct pollfd{
      • int fd;
      • short events;
      • short revents;
      • };

【Unix网络编程】chapter6IO复用的更多相关文章

  1. 【unix网络编程第三版】阅读笔记(五):I/O复用:select和poll函数

    本博文主要针对UNP一书中的第六章内容来聊聊I/O复用技术以及其在网络编程中的实现 1. I/O复用技术 I/O多路复用是指内核一旦发现进程指定的一个或者多个I/O条件准备就绪,它就通知该进程.I/O ...

  2. UNIX网络编程——套接字选项(心跳检测、绑定地址复用)

    /* 设置套接字选项周期性消息检测连通性 心跳包. 心博.主要用于长连接. * 参数:套接字, 1或0开启, 首次间隔时间, 两次间隔时间, 断开次数 */ void setKeepAlive( in ...

  3. UNIX网络编程 第6章 I/O复用:select和poll函数

    UNIX网络编程 第6章 I/O复用:select和poll函数

  4. Linux网络编程-IO复用技术

    IO复用是Linux中的IO模型之一,IO复用就是进程预先告诉内核需要监视的IO条件,使得内核一旦发现进程指定的一个或多个IO条件就绪,就通过进程进程处理,从而不会在单个IO上阻塞了.Linux中,提 ...

  5. Unix网络编程--卷一:套接字联网API

    UNIX网络编程--卷一:套接字联网API 本书面对的读者是那些希望自己编写的程序能够使用成为套接字(socket)的API进行彼此通信的人. 目录: 0.准备环境 1.简介 2.传输层:TCP.UD ...

  6. [转载] 读《UNIX网络编程 卷1:套接字联网API》

    原文: http://cstdlib.com/tech/2014/10/09/read-unix-network-programming-1/ 文章写的很清楚, 适合初学者 最近看了<UNIX网 ...

  7. UNIX网络编程——客户/服务器心搏函数

    阅读此博客时,可以参考以前的博客<<UNIX网络编程--socket的keep-alive>>和<<UNIX网络编程--套接字选项(心跳检测.绑定地址复用)> ...

  8. Unix网络编程中的五种I/O模型_转

    转自:Unix网络编程中的的五种I/O模型 下面主要是把unp第六章介绍的五种I/O模型. 1. 阻塞I/O模型 例如UDP函数recvfrom的内核到应用层.应用层到内核的调用过程是这样的:首先把描 ...

  9. UNIX网络编程——网络I/O模型

    在学习UNIX网络编程的时候.一開始分不清 同步 和 异步,所以还是总结一下,理清下他们的差别比較好. IO分类 IO依据对IO的调度方式可分为堵塞IO.非堵塞IO.IO复用.信号驱动IO.异步IO. ...

  10. 《Unix 网络编程》08:基本UDP套接字编程

    基本UDP套接字编程 系列文章导航:<Unix 网络编程>笔记 UDP 概述 流程图 recvfrom 和 sendto #include <sys/socket.h> ssi ...

随机推荐

  1. c/c++编译和链接过程

    编译器把一个cpp编译为目标文件的时候,除了要在目标文件里写入cpp里包含的数据和代码,还要至少提供3个表:未解决符号表,导出符号表和地址重定向表. 未解决符号表提供了所有在该编译单元里引用但是定义并 ...

  2. java BIO/NIO

    一.BIO Blocking IO(即阻塞IO); 1.      特点: a)   Socket服务端在监听过程中每次accept到一个客户端的Socket连接,就要处理这个请求,而此时其他连接过来 ...

  3. Qt开发问答

    Qt开发问答 1, Difference between Dialog and widget and QMainWindow http://www.qtcentre.org/threads/3465- ...

  4. node start - hello world http server

    Write a file t1.js 'use strict'; const express = require('express'); // Constants const PORT = 8080; ...

  5. getModifiers 方法解释。

    modifier: 字面意思修饰符. getModifiers  得到的就是 前面的 的修饰符 ,这个方法 字段和方法 都有.这个方法的值是 修饰符 相加的到的值. 例子: public class ...

  6. Hadoop "Cannot create directory .Name node is in safe mode."解决方案

    转载自:http://www.waitig.com/hadoop-name-node-is-in-safe-mode.html 在使用Hadoop建立文件的时候,出现“Cannot create di ...

  7. 使用 Kafka 在生产环境构建大规模机器学习

    智能实时应用为所有行业带来了革命性变化.机器学习及其分支深度学习正蓬勃发展,因为机器学习让计算机能够在无人指引的情况下挖掘深藏的洞见.这种能力正是多种领域所需要的,如非结构化数据分析.图像识别.语音识 ...

  8. MySQL Master High Available 源码篇

    https://m.aliyun.com/yunqi/users/1287368569594542/articles https://yq.aliyun.com/articles/59233 MySQ ...

  9. IIS7 UNC File caching issue

    You have to either choose dir-monitoring and file-change-notification with its drawback of using SMB ...

  10. Visual studio 下C++工程相关经验

    1.链接其他库调试时产生告警: warning LNK4099: 未找到 PDB“vc100.pdb” 解决方案:属性 -> C/C++ -> 输出文件 -> 程序数据库文件名 -& ...