我猜测你的实际代码长这样。
void new_connection (int sock) {
}
void accept_loop (int listen_sock) {
int new_sock;
while ((new_sock = accept(listen_sock, 0, 0)) != -1) {
std::thread t(new_connection, new_sock);
}
}
这段代码的问题在于当循环重新开始时,线程的析构函数被调用。由于析构函数会检测线程上下文仍然处于活动状态,因此会导致异常抛出。
为避免这个问题,您可以将线程对象从活动上下文中分离。
while ((new_sock = accept(listen_sock, 0, 0)) != -1) {
std::thread t(new_connection, new_sock);
t.detach();
}
以下是一个基本完整的示例(没有错误检查)。此程序创建一个接受套接字,用于服务器规范(对于特定接口,“主机:端口”,或对于任何接口,“:端口”)。
请注意保留HTML标记。
int make_accept_sock (const char *servspec) {
const int one = 1;
struct addrinfo hints = {};
struct addrinfo *res = 0, *ai = 0, *ai4 = 0;
char *node = strdup(servspec);
char *service = strrchr(node, ':');
int sock;
hints.ai_family = PF_UNSPEC;
hints.ai_socktype = SOCK_STREAM;
hints.ai_flags = AI_PASSIVE;
*service++ = '\0';
getaddrinfo(*node ? node : "0::0", service, &hints, &res);
free(node);
for (ai = res; ai; ai = ai->ai_next) {
if (ai->ai_family == PF_INET6) break;
else if (ai->ai_family == PF_INET) ai4 = ai;
}
ai = ai ? ai : ai4;
sock = socket(ai->ai_family, SOCK_STREAM, 0);
setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &one, sizeof(one));
bind(sock, ai->ai_addr, ai->ai_addrlen);
listen(sock, 256);
freeaddrinfo(res);
return sock;
}
接受循环程序创建监听套接字,然后启动线程来处理每个新的传入连接。
void accept_loop (const char *servspec) {
int sock = make_accept_sock(servspec);
for (;;) {
int new_sock = accept(sock, 0, 0);
std::thread t(new_connection, new_sock);
t.detach();
}
}
新的连接处理程序每秒钟输出一个点(
.
)和一个换行符。可以在
这个问题的答案中找到
isclosed()
函数。
void new_connection (int sock) {
ssize_t r;
while (!isclosed(sock)) {
r = send(sock, ".\n", 2, 0);
if (r < 0) break;
sleep(1);
}
close(sock);
}
然后主函数将所有内容结合在一起。
int main (int argc, char *argv[])
{
const char *server = ":11111";
signal(SIGPIPE, SIG_IGN);
if (argc > 1) server = argv[1];
accept_loop(server);
return 0;
}
select
函数。 - Captain Obvliouspoll
而不是select
,因为它的语义更容易理解。但是,你应该使用一些更高级别的接口或库,比如libevent
之类的东西。然而,线程应该是可以工作的,并且它是最容易让人理解的,所以请解释一下,附上一些示例代码,说明某些东西没有按照你的预期工作。 - jxh