【发布时间】:2011-07-16 00:17:38
【问题描述】:
我正在尝试在服务器和客户端之间进行非阻塞 I/O。两者连接后,我尝试使用 fork 处理 IO,但服务器端在尝试读取“传输端点未连接”时出错,并且发生两次(因为我猜是 fork?) .
服务器代码
//includes taken out
#define PORT "4950"
#define STDIN 0
struct sockaddr name;
void set_nonblock(int socket) {
int flags;
flags = fcntl(socket,F_GETFL,0);
assert(flags != -1);
fcntl(socket, F_SETFL, flags | O_NONBLOCK);
}
// get sockaddr, IPv4 or IPv6:
void *get_in_addr(struct sockaddr *sa) {
if (sa->sa_family == AF_INET)
return &(((struct sockaddr_in*)sa)->sin_addr);
return &(((struct sockaddr_in6*)sa)->sin6_addr);
}
int main(int agrc, char** argv) {
int status, sock, adrlen, new_sd;
struct addrinfo hints;
struct addrinfo *servinfo; //will point to the results
//store the connecting address and size
struct sockaddr_storage their_addr;
socklen_t their_addr_size;
//socket infoS
memset(&hints, 0, sizeof hints); //make sure the struct is empty
hints.ai_family = AF_INET;
hints.ai_socktype = SOCK_STREAM; //tcp
hints.ai_flags = AI_PASSIVE; //use local-host address
//get server info, put into servinfo
if ((status = getaddrinfo("127.0.0.1", PORT, &hints, &servinfo)) != 0) {
fprintf(stderr, "getaddrinfo error: %s\n", gai_strerror(status));
exit(1);
}
//make socket
sock = socket(servinfo->ai_family, servinfo->ai_socktype, servinfo->ai_protocol);
if (sock < 0) {
printf("\nserver socket failure %m", errno);
exit(1);
}
//allow reuse of port
int yes=1;
if (setsockopt(sock,SOL_SOCKET,SO_REUSEADDR,&yes,sizeof(int)) == -1) {
perror("setsockopt");
exit(1);
}
//unlink and bind
unlink("127.0.0.1");
if(bind (sock, servinfo->ai_addr, servinfo->ai_addrlen) < 0) {
printf("\nBind error %m", errno);
exit(1);
}
freeaddrinfo(servinfo);
//listen
if(listen(sock, 5) < 0) {
printf("\nListen error %m", errno);
exit(1);
}
their_addr_size = sizeof(their_addr);
//accept
new_sd = accept(sock, (struct sockaddr*)&their_addr, &their_addr_size);
if( new_sd < 0) {
printf("\nAccept error %m", errno);
exit(1);
}
cout<<"\nSuccessful Connection!";
//set nonblock
set_nonblock(new_sd);
char* in = new char[255];
char* out = new char[255];
int numSent;
int numRead;
pid_t pid;
fork();
pid = getpid();
if(pid == 0) {
while( !(out[0] == 'q' && out[1] == 'u' && out[2] == 'i' && out[3] == 't') ) {
fgets(out, 255, stdin);
numSent = send(sock, out, strlen(out), 0);
if(numSent < 0) {
printf("\nError sending %m", errno);
exit(1);
} //end error
} //end while
} //end child
else {
numRead = recv(sock, in, 255, 0);
if(numRead < 0) {
printf("\nError reading %m", errno);
exit(1);
} //end error
else {
cout<<in;
for(int i=0;i<255;i++)
in[i] = '\0';
} //end else
} //end parent
cout<<"\n\nExiting normally\n";
return 0;
}
客户端代码
//包括取出
#define PORT "4950"
struct sockaddr name;
void set_nonblock(int socket) {
int flags;
flags = fcntl(socket,F_GETFL,0);
assert(flags != -1);
fcntl(socket, F_SETFL, flags | O_NONBLOCK);
}
// get sockaddr, IPv4 or IPv6:
void *get_in_addr(struct sockaddr *sa) {
if (sa->sa_family == AF_INET)
return &(((struct sockaddr_in*)sa)->sin_addr);
return &(((struct sockaddr_in6*)sa)->sin6_addr);
}
int main(int agrc, char** argv) {
int status, sock, adrlen;
struct addrinfo hints;
struct addrinfo *servinfo; //will point to the results
memset(&hints, 0, sizeof hints); //make sure the struct is empty
hints.ai_family = AF_INET;
hints.ai_socktype = SOCK_STREAM; //tcp
hints.ai_flags = AI_PASSIVE; //use local-host address
//get server info, put into servinfo
if ((status = getaddrinfo("127.0.0.1", PORT, &hints, &servinfo)) != 0) {
fprintf(stderr, "getaddrinfo error: %s\n", gai_strerror(status));
exit(1);
}
//make socket
sock = socket(servinfo->ai_family, servinfo->ai_socktype, servinfo->ai_protocol);
if (sock < 0) {
printf("\nserver socket failure %m", errno);
exit(1);
}
if(connect(sock, servinfo->ai_addr, servinfo->ai_addrlen) < 0) {
printf("\nclient connection failure %m", errno);
exit(1);
}
cout<<"\nSuccessful connection!";
//set nonblock
set_nonblock(sock);
char* out = new char[255];
char* in = new char[255];
int numRead;
int numSent;
pid_t pid;
fork();
pid = getpid();
if(pid == 0) {
while( !(out[0] == 'q' && out[1] == 'u' && out[2] == 'i' && out[3] == 't') ) {
fgets(out, 255, stdin);
numSent = send(sock, out, strlen(out), 0);
if(numSent < 0) {
printf("\nError sending %m", errno);
exit(1);
} //end error
} //end while
} //end child process
else {
while( !(in[0] == 'q' && in[1] == 'u' && in[2] == 'i' && in[3] == 't') ) {
numRead = recv(sock, in, 255, 0);
cout<<in;
for(int i=0;i<255;i++)
in[i] = '\0';
}
} //end parent process
cout<<"\n\nExiting normally\n";
return 0;
}
我还尝试使用线程来执行 I/O。那里的问题是,当我运行程序时,就像线程没有发生一样。该程序仅以“成功连接”运行,然后“正常退出”。我将一些 cout 语句放入 while(1) 循环中,它们确实打印了几次,但它们只是由于某种原因停止了。我不确定这是我的线程还是我的套接字的问题。代码(与上面非常相似)在这里 -
服务器
//includes taken out
#define PORT "4950"
#define STDIN 0
pthread_t readthread;
pthread_t sendthread;
char* in = new char[255];
char* out = new char[255];
int numSent;
int numRead;
struct sockaddr name;
int sock, new_sd;
void* readThread(void* threadid) {
while(1) {
numRead = recv(new_sd, in, 255, 0);
if(numRead > 0) {
cout<<"\n"<<in;
for(int i=0;i<strlen(in);i++)
in[i] = '\0';
} //end if
else if(numRead < 0) {
printf("\nError reading %m", errno);
exit(1);
}
} //end while
} //END READTHREAD
void* sendThread(void* threadid) {
while(1) {
cin.getline(out, 255);
numSent = send(new_sd, out, 255, 0);
if(numSent > 0) {
for(int i=0;i<strlen(out);i++)
out[i] = '\0';
} //end if
else if(numSent < 0) {
printf("\nError sending %m", errno);
exit(1);
}
} //end while
} //END SENDTHREAD
void set_nonblock(int socket) {
int flags;
flags = fcntl(socket,F_GETFL,0);
assert(flags != -1);
fcntl(socket, F_SETFL, flags | O_NONBLOCK);
}
// get sockaddr, IPv4 or IPv6:
void *get_in_addr(struct sockaddr *sa) {
if (sa->sa_family == AF_INET)
return &(((struct sockaddr_in*)sa)->sin_addr);
return &(((struct sockaddr_in6*)sa)->sin6_addr);
}
int main(int agrc, char** argv) {
int status, adrlen;
struct addrinfo hints;
struct addrinfo *servinfo; //will point to the results
//store the connecting address and size
struct sockaddr_storage their_addr;
socklen_t their_addr_size;
//socket infoS
memset(&hints, 0, sizeof hints); //make sure the struct is empty
hints.ai_family = AF_INET;
hints.ai_socktype = SOCK_STREAM; //tcp
hints.ai_flags = AI_PASSIVE; //use local-host address
//get server info, put into servinfo
if ((status = getaddrinfo("127.0.0.1", PORT, &hints, &servinfo)) != 0) {
fprintf(stderr, "getaddrinfo error: %s\n", gai_strerror(status));
exit(1);
}
//make socket
sock = socket(servinfo->ai_family, servinfo->ai_socktype, servinfo->ai_protocol);
if (sock < 0) {
printf("\nserver socket failure %m", errno);
exit(1);
}
//allow reuse of port
int yes=1;
if (setsockopt(sock,SOL_SOCKET,SO_REUSEADDR,&yes,sizeof(int)) == -1) {
perror("setsockopt");
exit(1);
}
//unlink and bind
unlink("127.0.0.1");
if(bind (sock, servinfo->ai_addr, servinfo->ai_addrlen) < 0) {
printf("\nBind error %m", errno);
exit(1);
}
freeaddrinfo(servinfo);
//listen
if(listen(sock, 5) < 0) {
printf("\nListen error %m", errno);
exit(1);
}
their_addr_size = sizeof(their_addr);
//accept
new_sd = accept(sock, (struct sockaddr*)&their_addr, &their_addr_size);
if( new_sd < 0) {
printf("\nAccept error %m", errno);
exit(1);
}
cout<<"\nSuccessful Connection!";
//set nonblock
set_nonblock(new_sd);
pthread_attr_t attr;
pthread_attr_init(&attr);
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE);
pthread_create(&readthread, &attr, readThread, (void*)0);
pthread_create(&sendthread, &attr, sendThread, (void*)1);
cout<<"\n\nExiting normally\n";
return 0;
}
客户
#define PORT "4950"
pthread_t readthread;
pthread_t sendthread;
char* in = new char[255];
char* out = new char[255];
int numSent;
int numRead;
struct sockaddr name;
int sock;
void* readThread(void* threadid) {
while(1) {
numRead = recv(sock, in, 255, 0);
if(numRead > 0) {
cout<<"\n"<<in;
for(int i=0;i<strlen(in);i++)
in[i] = '\0';
} //end if
else if(numRead < 0) {
printf("\nError reading %m", errno);
exit(1);
}
} //end while
} //END READTHREAD
void* sendThread(void* threadid) {
while(1) {
cin.getline(out, 255);
numSent = send(sock, out, 255, 0);
if(numSent > 0) {
for(int i=0;i<strlen(out);i++)
out[i] = '\0';
} //end if
else if(numSent < 0) {
printf("\nError sending %m", errno);
exit(1);
}
} //end while
} //END SENDTHREAD
void set_nonblock(int socket) {
int flags;
flags = fcntl(socket,F_GETFL,0);
assert(flags != -1);
fcntl(socket, F_SETFL, flags | O_NONBLOCK);
}
// get sockaddr, IPv4 or IPv6:
void *get_in_addr(struct sockaddr *sa) {
if (sa->sa_family == AF_INET)
return &(((struct sockaddr_in*)sa)->sin_addr);
return &(((struct sockaddr_in6*)sa)->sin6_addr);
}
int main(int agrc, char** argv) {
int status, adrlen;
struct addrinfo hints;
struct addrinfo *servinfo; //will point to the results
memset(&hints, 0, sizeof hints); //make sure the struct is empty
hints.ai_family = AF_INET;
hints.ai_socktype = SOCK_STREAM; //tcp
hints.ai_flags = AI_PASSIVE; //use local-host address
//get server info, put into servinfo
if ((status = getaddrinfo("127.0.0.1", PORT, &hints, &servinfo)) != 0) {
fprintf(stderr, "getaddrinfo error: %s\n", gai_strerror(status));
exit(1);
}
//make socket
sock = socket(servinfo->ai_family, servinfo->ai_socktype, servinfo->ai_protocol);
if (sock < 0) {
printf("\nserver socket failure %m", errno);
exit(1);
}
if(connect(sock, servinfo->ai_addr, servinfo->ai_addrlen) < 0) {
printf("\nclient connection failure %m", errno);
exit(1);
}
cout<<"\nSuccessful connection!";
//set nonblock
set_nonblock(sock);
pthread_attr_t attr;
pthread_attr_init(&attr);
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE);
pthread_create(&readthread, &attr, readThread, (void*)0);
pthread_create(&sendthread, &attr, sendThread, (void*)1);
cout<<"\n\nExiting normally\n";
return 0;
}
我为这篇冗长的帖子道歉,但我已经在这几天了,不知道如何继续。我尝试使用 select(),但对于 1 个客户端和一个服务器 I/O 来说,这似乎有点多。如果有人能指出上面可能出现的问题或任何其他提示(或者我只是对 select 完全错误 :)),我将不胜感激。
【问题讨论】:
-
如果您喜欢阅读源代码,请查看 node.js 项目。他们在 IO 方面做得非常好,而且一切都有很好的记录。你可以借用一些类似的概念,或者直接拿代码,它是 MIT。