linux教程网络高级编程.ppt
《linux教程网络高级编程.ppt》由会员分享,可在线阅读,更多相关《linux教程网络高级编程.ppt(21页珍藏版)》请在三一文库上搜索。
1、I/O模式,五种 I/O操作方式: 阻塞 I/O 非阻塞 I/O I/O多路复用 信号驱动 I/O(SIGIO) 异步 I/O 一般来说,程序进行输入操作有两步: 1等待有数据可以读 2将数据从系统内核中拷贝到程序的数据区。 对于一个对套接字的输入操作,第一步一般来说是等待数据从网络上传到本地。当数据包到达的时候,数据将会从网络层拷贝到内核的缓存中;第二步是从内核中把数据拷贝到程序的数据区中。,阻塞 I/O模式,阻塞 I/O模式是最普遍使用的 I/O模式。大部分程序使用的都是阻塞模式的 I/O 。缺省的,一个套接字建立后所处于的模式就是阻塞 I/O模式。 对于一个 UDP套接字来说,数据就绪的
2、标志比较简单: 已经收到了一整个数据报 没有收到。 而 TCP这个概念就比较复杂,需要附加一些其他的变量。一个进程调用 recvfrom ,然后系统调用并不返回直到有数据报到达本地系统,然后系统将数据拷贝到进程的缓存中。(如果系统调用收到一个中断信号,则它的调用会被中断)我们称这个进程在调用 recvfrom一直到从 recvfrom返回这段时间是阻塞的。当 recvfrom正常返回时,我们的进程继续它的操作。,非阻塞模式 I/O,当我们将一个套接字设置为非阻塞模式,我们相当于告诉了系统内核:“当我请求的 I/O操作不能够马上完成,你想让我的进程进行休眠等待的时候,不要这么做,请马上返回一个错
3、误给我。” 我们开始对 recvfrom的三次调用,因为系统还没有接收到网络数据,所以内核马上返回一个 EWOULDBLOCK的错误。第四次我们调用 recvfrom函数,一个数据报已经到达了,内核将它拷贝到我们的应用程序的缓冲区中,然后 recvfrom正常返回,我们就可以对接收到的数据进行处理了。 当一个应用程序使用了非阻塞模式的套接字,它需要使用一个循环来不听的测试是否一个文件描述符有数据可读(称做 polling)。应用程序不停的 polling内核来检查是否I/O操作已经就绪。这将是一个极浪费 CPU资源的操作。这种模式使用中不是很普遍。,I/O多路复用,在使用 I/O多路技术的时候
4、,我们调用 select()函数和 poll()函数的时候阻塞,而不是调用 recvfrom(或 recv)的时候阻塞。 当我们调用 select函数阻塞的时候,select函数等待数据报套接字进入读就绪状态。当 select函数返回的时候,也就是套接字可以读取数据的时候。这时候我们就可以调用recvfrom函数来将数据拷贝到我们的程序缓冲区中。 和阻塞模式相比较,select()和 poll()并没有什么高级的地方,而且,在阻塞模式下只需要调用一个函数:读取或发送,在使用了多路复用技术后,我们需要调用两个函数了:先调用 select()函数或 poll()函数,然后才能进行真正的读写。 多路
5、复用的高级之处在于,它能同时等待多个文件描述符,而这些文件描述符(套接字描述符)其中的任意一个进入读就绪状态,select()函数就可以返回。 假设我们运行一个网络客户端程序,要同时处理套接字传来的网络数据又要处理本地的标准输入输出。在我们的程序处于阻塞状态等待标准输入的数据的时候,假如服务器端的程序被 kill(或是自己 Down掉了),那么服务器程端的 TCP协议会给客户端(我们这端)的 TCP协议发送一个 FIN数据代表终止连接。但是我们的程序阻塞在等待标准输入的数据上,在它读取套接字数据之前(也许是很长一段时间),它不会看见结束标志我们就不能够使用阻塞模式的套接字。 IO多路技术一般在
6、下面这些情况中被使用: 当一个客户端需要同时处理多个文件描述符的输入输出操作的时候(一般来说是标准的输入输出和网络套接字), I/O多路复用技术将会有机会得到使用。 当程序需要同时进行多个套接字的操作的时候。 如果一个 TCP服务器程序同时处理正在侦听网络连接的套接字和已经连接好的套接字。 如果一个服务器程序同时使用 TCP和 UDP协议。 如果一个服务器同时使用多种服务并且每种服务可能使用不同的协议(比如 inetd就是这样的)。 I/O多路服用技术并不只局限与网络程序应用上。几乎所有的程序都可以找到应用 I/O多路复用的地方。,信号驱动 I/O模式,我们可以使用信号,让内核在文件描述符就绪
7、的时候使用 SIGIO信号来通知我们。我们将这种模式称为信号驱动 I/O模式。 使用这种模式,我们首先需要允许套接字使用信号驱动 I/O ,还要安装一个 SIGIO的处理函数。在这种模式下,系统调用将会立即返回,然后我们的程序可以继续做其他的事情。当数据就绪的时候,系统会向我们的进程发送一个 SIGIO信号。这样我们就可以在 SIGIO信号的处理函数中进行 I/O操作(或是我们在函数中通知主函数有数据可读)。 对于信号驱动 I/O模式,它的先进之处在于它在等待数据的时候不会阻塞,程序可以做自己的事情。当有数据到达的时候,系统内核会向程序发送一个 SIGIO信号进行通知,这样我们的程序就可以获得
8、更大的灵活性,因为我们不必为等待数据进行额外的编码。 信号 I/O可以使内核在某个文件描述符发生改变的时候发信号通知我们的程序。异步 I/O可以提高我们程序进行 I/O读写的效率。通过使用它,当我们的程序进行 I/O操作的时候,内核可以在初始化 I/O操作后立即返回,在进行 I/O操作的同时,我们的程序可以做自己的事情,直到 I/O操作结束,系统内核给我们的程序发消息通知。 基于 Berkeley接口的 Socket信号驱动 I/O使用信号 SIGIO。有的系统 SIGPOLL信号,它也是相当于 SIGIO的。 为了在一个套接字上使用信号驱动 I/O操作,下面这三步是所必须的。 (1)一个和
9、SIGIO信号的处理函数必须设定。 (2)套接字的拥有者必须被设定。一般来说是使用 fcntl函数的 F_SETOWN参数来进行设定拥有者。 (3)套接字必须被允许使用异步 I/O。一般是通过调用 fcntl函数的 F_SETFL命令, O_ASYNC为参数来实现。 注意:我们在设置套接字的属主之前必须将 SIGIO的信号处理函数设好,SIGIO的缺省动作是被忽略。因此我们如果以相反的顺序调用这两个函数调用,那么在 fcntl函数调用之后,signal函数调用之前就有一小段时间程序可能接收到 SIGIO信号。那样的话,信号将会被丢弃。在 SVR4系统中,SIGIO在 头文件中被定义为 SIGP
10、OLL,而 SIGPOLL信号的缺省动作是终止这个进程。所以我们一定要保证这两个函数的调用顺序:先调用 signal设置好 SIGIO信号处理函数,然后在使用 fcntl函数设置套接字的属主。,信号驱动 I/O模式,虽然设定套接字为异步 I/O非常简单,但是使用起来困难的部分是怎样在程序中断定产生 SIGIO信号发送给套接字属主的时候,程序处在什么状态。 1UDP套接字的 SIGIO信号在 UDP协议上使用异步 I/O非常简单这个信号将会在这个时候产生: 套接字收到了一个数据报的数据包。 套接字发生了异步错误。 当我们在使用 UDP套接字异步 I/O的时候,我们使用 recvfrom()函数来
11、读取数据报数据或是异步 I/O错误信息。 2TCP套接字的 SIGIO信号 不幸的是,异步 I/O几乎对 TCP套接字而言没有什么作用。因为对于一个 TCP套接字来说, SIGIO信号发生的几率太高了,所以 SIGIO信号并不能告诉我们究竟发生了什么事情。在 TCP连接中, SIGIO信号将会在这个时候产生: 在一个监听某个端口的套接字上成功的建立了一个新连接。 一个断线的请求被成功的初始化。 一个断线的请求成功的结束。 套接字的某一个通道(发送通道或是接收通道)被关闭。 套接字接收到新数据。 套接字将数据发送出去。 发生了一个异步 I/O的错误。 举例来说,如果一个正在进行读写操作的 TCP
12、套接字处于信号驱动 I/O状态下,那么每当新数据到达本地的时候,将会产生一个 SIGIO信号,每当本地套接字发出的数据被远程确认后,也会产生一个 SIGIO信号。对于我们的程序来讲,是无法区分这两个 SIGIO有什么区别的。在这种情况下使用 SIGIO,TCP套接字应当被设置为无阻塞模式来阻止一个阻塞的 read和 write(recv和 send)操作。我们可以考虑在一个只进行监听网络连接操作的套接字上使用异步 I/O,这样当有一个新的连接的时候,SIGIO信号将会产生。,异步 I/O模式,当我们运行在异步 I/O模式下时,我们如果想进行 I/O操作,只需要告诉内核我们要进行 I/O操作,然
13、后内核会马上返回。具体的 I/O和数据的拷贝全部由内核来完成,我们的程序可以继续向下执行。当内核完成所有的 I/O操作和数据拷贝后,内核将通知我们的程序。 异步 I/O和信号驱动 I/O的区别是: 信号驱动 I/O模式下,内核在操作可以被操作的时候通知给我们的应用程序发送 SIGIO消息。 异步 I/O模式下,内核在所有的操作都已经被内核操作结束之后才会通知我们的应用程序。 当我们进行一个 IO操作的时候,我们传递给内核我们的文件描述符,我们的缓存区指针和缓存区的大小,一个偏移量 offset,以及在内核结束所有操作后和我们联系的方法。这种调用也是立即返回的,我们的程序不需要阻塞住来等待数据的
14、就绪。我们可以要求系统内核在所有的操作结束后(包括从网络上读取信息,然后拷贝到我们提供给内核的缓存区中)给我们发一个消息。,socket编程,socket() 函数,#include #include int socket(int domain , int type , int protocol); domain需要被设置为 “AF_INET” type参数告诉内核这个 socket是什么类型,“SOCK_STREAM”或是“ SOCK_DGRAM”。 把 protocol设置为 0 。 套接字创建时没有指定名字客户机用套接字的名字读写它。socket()函数只是简单的返回一个你以后可以使用的
15、套接字描述符。如果发生错误, socket()函数返回 1 。全局变量 errno将被设置为错误代码。,bind() 函数,#include #include int bind (int sockfd , struct sockaddr *my_addr , int addrlen) ; 参数说明: sockfd是由 socket()函数返回的套接字描述符。 my_addr是一个指向 struct sockaddr的指针。 addrlen可以设置为 sizeof(struct sockaddr)。 bind()函数可以帮助你指定一个套接字使用的端口。 当你需要进行端口监听 listen()操作
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- linux 教程 网络 高级 编程
链接地址:https://www.31doc.com/p-2976963.html