gpt4 book ai didi

c - Unix 网络编程说明

转载 作者:可可西里 更新时间:2023-11-01 02:39:15 26 4
gpt4 key购买 nike

我在看经典书Unix Network Programming ,当我偶然发现这个程序时(第 6.8 节,第 179-180 页)

#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, 0);

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 = -1; /* index into client[] array */
for (i = 0; i < FD_SETSIZE; i++)
client[i] = -1; /* -1 indicates available entry */
FD_ZERO(&allset);
FD_SET(listenfd, &allset);

for ( ; ; ) {
rset = allset; /* structure assignment */
nready = Select(maxfd+1, &rset, NULL, NULL, NULL);

if (FD_ISSET(listenfd, &rset)) { /* new client connection */
clilen = sizeof(cliaddr);
connfd = Accept(listenfd, (SA *) &cliaddr, &clilen);

for (i = 0; i < FD_SETSIZE; i++)
if (client[i] < 0) {
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 <= 0)
continue; /* no more readable descriptors */
}

for (i = 0; i <= maxi; i++) { /* check all clients for data */
if ( (sockfd = client[i]) < 0)
continue;
if (FD_ISSET(sockfd, &rset)) {
if ( (n = Read(sockfd, buf, MAXLINE)) == 0) {
/*4connection closed by client */
Close(sockfd);
FD_CLR(sockfd, &allset);
client[i] = -1;
} else
Writen(sockfd, buf, n);

if (--nready <= 0)
break; /* no more readable descriptors */
}
}
}
}

作者提到这个程序不能抵御 DOS 攻击。引自书中,

“不幸的是,我们刚刚展示的服务器存在问题。考虑一下如果恶意客户端连接到服务器,发送一个字节的数据(换行符除外),然后进入休眠状态会发生什么情况. 服务端会调用read(系统调用),从客户端读取单字节数据,然后阻塞在下一次调用read,等待更多来自这个客户端的数据。然后服务器被这个客户端阻塞,并且不会为任何其他客户端提供服务,直到恶意客户端发送换行符或终止“

我不确定我是否理解正确。为什么这个恶意客户端会第二次调用 read 系统调用,因为它只发送了一个字节的数据,它会在第一次调用 select 时得到通知。由于没有事件,随后对 select 的调用将永远不会设置此恶意文件描述符。我在这里遗漏了什么吗?

这里我猜测是代码有错别字,不是Read,应该是书中其他地方提到的某个版本的Readline方法。

注意:代码包含ReadSelect(大写R和S​​),它们只不过是的错误处理包装器>阅读选择系统调用

最佳答案

是的,它似乎有意成为 Readline

downloadable source code该文件是 tcpcliserv/tcpservselect01.c 并且有一个相应的 .lc 文件(带有行号注释),它使用 Readline 而不是 Read,在本书第二版(source code)中是Readline。理解括号注释“(换行符除外)”的唯一方法是假设预期的读取函数读取到换行符。

奇怪的是,它没有被报道in the errata .也许你应该这样做。

关于c - Unix 网络编程说明,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31064833/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com