Chinaunix
标题:
keepalive持久连接时,怎么知道结束的
[打印本页]
作者:
zhongyj
时间:
2013-10-16 12:46
标题:
keepalive持久连接时,怎么知道结束的
浏览器和Web服务器之间不是持久连接时,Web服务器传输完后会调用close,这时浏览器的read会返回。
进行持久连接时,因为Web服务器没有调用close,那浏览器会不会一直阻塞在read不能返回呢;如果浏览器使用非阻塞read,是不是会存在只读了一半文件的情况。
作者:
myworkstation
时间:
2013-10-16 16:04
本帖最后由 myworkstation 于 2013-10-16 16:05 编辑
回复
1#
zhongyj
针对HTTP连接而言无论服务器还是客户端都不会让一个空闲的连接一直存在,都有相应的默认超时时间,超过这个时间就会断开连接。至于read阻塞问题纯属程序实现的部分(为什么要阻塞),正常的请求响应不会在read到一半时被服务器主动关闭的,除非网络有问题或者服务器挂掉。keep-alive在HTTP1.0和HTTP1.1中有区别,但是无论如何一个连接的关闭是会反应到socket的状态变化上的。怎么就不能知道了呢。
作者:
zhongyj
时间:
2013-10-17 10:05
我用代码来说明一下问题吧。这样清楚些。
服务端
int fileFd = open(argv[1], O_RDONLY);
for(;;)
{
int socketFd = accept(listenfd, NULL, NULL)
ssize_t len;
char readBuf[1024];
while(len = read(socketFd, readBuf, 1024) > 0) // 只要客户端发送一次数据,就把一个静态文件返回
{
char buffer[BUF_SIZE];
ssize_t size;
while((size = read(fileFd, buffer, BUF_SIZE)) > 0)
{
write(socketFd, buffer, size);
}
lseek(fileFd, 0, SEEK_SET);
}
close(socketFd);
}
复制代码
客户端
int count = 1000;
while(count--) // 做1000次请求
{
write(sockfd, "client", 6);
int readLength;
while((readLength = read(sockfd, rcvBuff, BUF_SIZE)) > 0)
{
// 我什么时候break?
}
}
复制代码
客户端连接服务端后,做1000次请求
客户端发送一次请求(write)后就循环read获取结果,但是不知道什么时候推出循环。
如果不是keepAlive的话,服务端发送往文件后会执行close,这时while中的read会返回-1,这时循环就退出了。
但在keepAlive情况下,服务端现在写完后就read等待客户端发送数据了。
客户端如果采用阻塞read,就会一直阻塞在read。
客户端如果采用非阻塞read,我觉得可能出现这种情况,比如服务端write写了两块数据给客户端,客户端第一次read完了,但是第二块数据还没到达客户端的时候客户端执行read了,这时因为是非阻塞就立即返回了。也就是说服务端发送了整个文件的数据,但是因为客户端使用非阻塞只读取了部分数据。
回复
2#
myworkstation
作者:
myworkstation
时间:
2013-10-17 10:54
回复
3#
zhongyj
文件长度信息呢?HTTP协议中Content-Length用于指定内容大小。完全没有看到你有处理的意思啊。
作者:
folklore
时间:
2013-10-17 11:51
你只读content-length长度的数据, 它不就返回了么?
作者:
linux_c_py_php
时间:
2013-10-17 13:48
读了部分就读了部分吧, 有什么问题?
作者:
zhongyj
时间:
2013-10-17 15:15
TKS,就是这个了。
回复
4#
myworkstation
作者:
sxcong
时间:
2013-10-23 10:47
排除其他情况,浏览器会有个请求超时的处理的。
欢迎光临 Chinaunix (http://bbs.chinaunix.net/)
Powered by Discuz! X3.2