この質問が以前に聞かれた場合はお詫び申し上げます。私は選択多重化を使用して非ブロッキングソケットクライアントを作成しています。私を混乱させることの1つは、オンラインまたはオフラインになっているサーバーに関係なく、非ブロッキング接続が常に成功することです。私は多くの記事を検索し、彼らのソリューションに従ったが、それらのどれも私のLinuxのubuntuマシンでは動作しません。ノンブロッキングソケット接続は常に成功しますか?
毎回static void callback_on_select_write(int connect_fd) {
// Client write event arrived;
int error = -1;
socklen_t len = sizeof(error);
if(getsockopt(connect_fd, SOL_SOCKET, SO_ERROR, &error, &len) == -1) {
return;
}
// getsockopt puts the errno value for connect into erro so 0 means no-error.
if(error == 0) {
// Connection ok.
}
else {
cerr << "Failed to connect\n";
return;
}
// Ready to write/read
}
選択戻り、代わりに常に失敗をcerringの、ブロックを「読み/書きする準備ができました」に行く、すなわちを成功このコールバックを呼び出します。なぜこれが起こるのですか?接続が本当に成功したかどうかを検出するポータブルメカニズムを設計するにはどうすればよいですか?以下は、私がコネクタを作成する方法です。毎回
int make_socket_client(const ::std::string& hostname, const ::std::string& port) {
struct addrinfo hints;
struct addrinfo* res {nullptr};
struct addrinfo* ptr {nullptr};
memset(&hints, 0, sizeof(struct addrinfo));
hints.ai_family = AF_UNSPEC;
hints.ai_socktype = SOCK_STREAM;
hints.ai_protocol = IPPROTO_TCP;
int rv;
int connector;
if((rv = getaddrinfo(hostname.c_str(), port.c_str(), &hints, &res)) != 0) {
return -1;
}
// Try to get the first available client connection.
for(ptr = res; ptr != nullptr; ptr = ptr->ai_next) {
// Ignore undefined ip type.
if(ptr->ai_family != AF_INET && ptr->ai_family != AF_INET6) {
continue;
}
// Create a listener socket and bind it to the localhost as the server.
if((connector = socket(ptr->ai_family, ptr->ai_socktype, ptr->ai_protocol)) == -1){
continue;
}
make_fd_nonblocking(connector);
if(connect(connector, (struct sockaddr*)ptr->ai_addr, ptr->ai_addrlen) < 0) {
// This is what we expect.
if(errno == EINPROGRESS) {
break;
}
else {
close(connector);
continue;
}
}
else {
break;
}
}
freeaddrinfo(res);
if(ptr == nullptr) {
return -1;
}
return connector;
}
確かにあなたは 'connect_fd'のSO_ERRORを見ているべきですか? 「イベント」とは何ですか?このコードは、質問に答えるのに十分ではありません。 – EJP
申し訳ありません。それはconnect_fdのみでなければなりません。 – Jes
元の 'connect()'はゼロを返しましたか? – EJP