Linux:对TCP阻塞控制/面向字节流/异常的理解
    		       		warning:
    		            这篇文章距离上次修改已过447天,其中的内容可能已经有所变动。
    		        
        		                
                在Linux中,TCP的块模式和非阻塞模式是两种常见的I/O模型。
- 阻塞I/O(blocking I/O):默认情况下,所有的套接字都是阻塞的。当进程调用一个阻塞的I/O函数时,该进程会被挂起,直到有数据可供处理。
 - 非阻塞I/O(nonblocking I/O):通过设置套接字选项为非阻塞,进程可以直接调用recvfrom()等函数,如果没有数据可读,这些函数会立即返回一个EWOULDBLOCK错误,而不会挂起进程。
 - TCP字节流(TCP stream):TCP作为一种字节流协议,提供了一种可靠的、面向连接的数据传输服务。
 - TCP异常(TCP exceptions):TCP异常指的是TCP协议中的一些特殊情况,如连接断开、网络超时等。
 
对于TCP异常的处理,可以使用select()或poll()系统调用,它们可以等待多个文件描述符上的某种事件,如果任何一个文件描述符上的事件发生,select()或poll()就会返回。这样,你可以检查哪个socket或文件描述符可以进行无阻塞的I/O操作。
以下是一个使用select()处理TCP异常的简单示例:
#include <sys/select.h>
#include <sys/time.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdio.h>
 
int main() {
    fd_set rfds;
    struct timeval tv;
    int retval;
 
    // 清除文件描述符集合
    FD_ZERO(&rfds);
 
    // 添加你想要检查的文件描述符到集合中
    FD_SET(0, &rfds); // 标准输入
    FD_SET(sockfd, &rfds); // 你的socket文件描述符
 
    // 超时设置
    tv.tv_sec = 1; // 秒
    tv.tv_usec = 0; // 微秒
 
    // 调用select()
    retval = select(sockfd + 1, &rfds, NULL, NULL, &tv);
 
    if (retval == -1) {
        // 错误处理
        perror("select()");
        exit(1);
    } else if (retval) {
        // 如果retval非零,则至少有一个描述符的事件发生了
        if (FD_ISSET(sockfd, &rfds)) {
            // 你的socket文件描述符上的事件
            // 可能是可读、可写或异常
            // 对于异常,你可能需要调用getsockopt()来检查
            // SOCKET_ERROR来获取错误代码
        }
    } else {
        // 超时处理
        printf("select() timed out.\n");
    }
 
    return 0;
}在这个例子中,select()会等待数据在标准输入或者指定的socket上可读、可写或者发生异常。如果在指定时间内没有任何事件发生,select()会超时返回。如果发生异常,你可能需要通过getsockopt()函数和SO\_ERROR选项来检查具体的错误代码。
评论已关闭