我的 TCP/IP 客户端在写入套接字时挂起。即使服务器通过close()
(或shutdown()
)调用正确关闭了已接受的连接,也会发生这种情况。对于这种情况,我一直认为 write 应该返回 ECONNRESET 错误。
如何防止同步输出挂断?或者更确切地说,我做错了什么以致不报告错误write()
?
我应该使用send()
代替write()
还是它们可以互换?
我正在使用两个线程在单独的应用程序中测试网络。主线程启动服务器线程,接受连接并立即关闭它。然后主线程通过连接到侦听端口来模仿客户端行为。
主线程代码:
sockaddr_in serv_addr;
bzero((char *) &serv_addr, sizeof(serv_addr));
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = INADDR_ANY;
serv_addr.sin_port = htons(port);
int s = socket(AF_INET, SOCK_STREAM, 0);
if (bind(s, (sockaddr*)(&serv_addr), sizeof(serv_addr)) < 0) {
close(s);
throw runtime_error(strerror(errno));
}
listen(s,1);
start_thread(s); // posix thread accepting a connection on s is started here
//Code below imitates TCP/IP client
struct hostent *hp;
struct sockaddr_in addr;
if((hp = gethostbyname(host.c_str())) == NULL){
throw HErrno();
}
bcopy(hp->h_addr, &addr.sin_addr, hp->h_length);
addr.sin_port = htons(port);
addr.sin_family = AF_INET;
int _socket = ::socket(PF_INET, SOCK_STREAM, IPPROTO_TCP);
if (_socket < 0) {
throw Errno();
}
assert(ENOTCONN==107);
assert(_socket>=0);
if(::connect(_socket, (const struct sockaddr *)&addr, sizeof(struct sockaddr_in)) == -1){
throw Errno();
}
while(true) {
const char a[] = "pattern";
if (write( _socket, a, 7)<0) // Writes 30000 times, then hangs
break;
}
服务器线程代码:
int connection = accept(s);
close(connection);
编辑: 问题减少到我的编程错误。看来我未能正确开始接受线程。