linux下accept之前异常终止连接问题?
服务端代码:
#include <sys/socket.h>
#include <unistd.h>
#include <sys/types.h>
#include <stdint.h>
#include <assert.h>
#include <fcntl.h>
#include <string.h>
#include <bits/socket.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <errno.h>
const char* ip = "192.168.16.105";
uint16_t port = 8888;
char buff[1024];
void make_sock_noblocking(int sock_fd)
{
int flags = fcntl(sock_fd, F_GETFL, 0);
flags |= O_NONBLOCK;
fcntl(sock_fd, F_SETFL, flags);
}
int main()
{
int listen_fd = socket(PF_INET, SOCK_STREAM, 0);
struct sockaddr_in sock_addr;
memset(&sock_addr, 0, sizeof(sock_addr));
sock_addr.sin_family = AF_INET;
sock_addr.sin_port = htons(port);
inet_pton(AF_INET, ip, &(sock_addr.sin_addr));
int optval = 1;
setsockopt(listen_fd, SOL_SOCKET, SO_REUSEADDR,
(const void *)&optval , sizeof(int));
int ret = bind(listen_fd, (sockaddr*)&sock_addr, sizeof(sock_addr));
listen(listen_fd, 1024);
make_sock_noblocking(listen_fd);
sleep(10);
struct sockaddr_in client_addr;
memset(&sock_addr, 0, sizeof(client_addr));
socklen_t len = sizeof(client_addr);
int conn_fd = accept(listen_fd, (sockaddr*)&client_addr, &(len));
printf("%d--------%s\n", conn_fd, strerror(errno));
int read_bytes = read(conn_fd, buff, sizeof(buff));
printf("%d %s\n", read_bytes, strerror(errno));
return 0;
}
客户端代码:
#include <sys/socket.h>
#include <unistd.h>
#include <sys/types.h>
#include <stdint.h>
#include <assert.h>
#include <fcntl.h>
#include <string.h>
#include <bits/socket.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <errno.h>
const char* ip = "192.168.16.105";
uint16_t port = 8888;
int main()
{
int sock_fd = socket(PF_INET, SOCK_STREAM, 0);
struct sockaddr_in sock_addr;
memset(&sock_addr, 0, sizeof(sock_addr));
sock_addr.sin_family = AF_INET;
sock_addr.sin_port = htons(port);
inet_pton(AF_INET, ip, &(sock_addr.sin_addr));
int ret = connect(sock_fd, (sockaddr*)&sock_addr, sizeof(sock_addr));
if(ret < 0)
{
printf("%s\n", strerror(errno));
}
struct linger my_linger;
memset(&my_linger, 0, sizeof(my_linger));
my_linger.l_onoff = 1;
my_linger.l_linger = 0;
setsockopt(sock_fd, SOL_SOCKET, SO_LINGER, &my_linger, sizeof(my_linger));
close(sock_fd);
return 0;
}
问题:
accpet之前sleep 10秒钟,这期间客户端向服务端发送RST终止连接,但是服务端这边还是会成功接受连接。按unp讲的不应该是返回一个ECONNABORTED(POSIX.1)错误吗?如果我把套接字的非阻塞去掉变成阻塞的话,依然可以成功接受连接,accept并不会阻塞,但是accept取出的这个socket去读的话就会出现Connection reset by peer错误了。莫非linux的实现是不管accept之前是不是被RST了,只管取出来,但取出来的这个socket再操作的话就会报错?
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(2)
在listen之后,系统就会维护相应的队列,处理所有对这个监听socket的连接,accept则是将已完成的连接从队列中取出来。因此不管你是否调用accept,来至client的tcp连接都是可以完成三次握手的。
根据你的代码,首先server端开始监听,然后进入sleep,这期间运行client,首先调用connect完成三次握手,这时候server端就建立了一个已完成连接并放入队列中,client再调用close关闭连接,发送FIN,相信应该是收到了ACK,因为TCP是全双工协议,因此server需要自己去关闭一个连接,client只是关闭了自己的连接,server的连接还是存在的。所以你可以accept获取到连接的,但接下来对这个socket进行操作就会返回错误,需要主动进行关闭。
之前没注意是使用RST中止连接的,看了一下2.6.32的内核代码,server在收到RST之后,会将所有状态清理,直接断掉连接,但是没有将申请的资源回收,这个过程是在close中完成的。所以accept时,监听队列中的结构还是存在可以取出,accept本身确实只是一个取的过程。
不管是否sleep10,客户端阻塞于connect,等到服务端来读套节字的时候客户端已经关闭了,然后报错connection reset by peer