3

我正在尝试编写一个实验性的客户端/服务器程序来证明当发送缓冲区已满时写入是否失败或阻塞。基本上,我在发送方程序上有一个无限循环,我使用 select() 检查是否可以在缓冲区上写入(我认为这意味着套接字缓冲区未满),如果我可以在缓冲区上写入而不是我 write() 一个字符。当 FD_ISSET(sockfd, &writefds) 为假时循环中断(我无法在缓冲区上写入,因为它已满)。接收程序在开始 read() 之前会休眠一分钟。我希望发送者在这段睡眠时间内填充缓冲区,但实际上,程序永远不会结束。

发件人:

int main(int argc, char *argv[]) {
    char buffer[100];
    int sockfd, total = 0, bytes = 0;
    fd_set writefds;

    sockfd = dial(argv[1], argv[2]);
    bzero(buffer, sizeof buffer);

    while(1)
    {
        int ret = 0;
        FD_ZERO(&writefds);
        FD_SET(sockfd, &writefds);

        if((ret = select(sockfd + 1, NULL, &writefds, NULL, 0)) < 0)
        {
            perror("select");
            exit(errno);
        }

        if(FD_ISSET(sockfd, &writefds))
        {
            write(sockfd, "a", 1);
            total++;
            continue;
        }
        else
        {
            puts("I can't write in the socket buffer");
            break;
        }
    }
    printf("nb chars written: %d\n", total);

    return 0;
}

接收者:

int foo(int sockfd) {
    char buffer[100];
    int t, total = 0;

    bzero(buffer, sizeof buffer);
    printf("I have a new client\n");

    sleep(60);

    while((t = read(sockfd, buffer, sizeof buffer)) > 0)
    {
        total += t;
        printf("%d ", total);
    }
    printf("nb chars read: %d\n", total);

    if(t < 0)
    {
        perror("read");
    }

    printf("I don't have that client anymore\n");
    return 0;
}
4

2 回答 2

0

您的选择超时为空,因此select()当发送缓冲区已满时会阻塞。这意味着当它返回时,套接字是可写的,并且你永远不会得到你的代码“我不能在套接字缓冲区中写入”。

参见手册页http://linux.die.net/man/2/select

如果你想要一个零超时,即不要阻塞select(),你需要将一个指针传递给一个 timeval 结构,并将两个字段都设置为零。

于 2014-10-20T08:34:39.277 回答
0

你在正确的轨道上,但套接字发送缓冲区可能是 48k 或更多。这是很多迭代。尝试一次写入 8k,而不仅仅是一个字节。并增加接收器读取之前的时间。

NB 没有真正需要测试这个。它在阻塞模式下阻塞,并在非阻塞模式下以 EAGAIN/EWOULDBLOCK 失败。请参阅手册页。

于 2014-03-31T12:10:17.140 回答