服务器打印到标准输出,而不是套接字 [英] Server prints to stdout instead of socket
问题描述
我正在学习UNIX网络编程中的示例,并且我已经将"daytimeclientserv.c"改编成这里的代码。服务器按照预期将日期/时间字符串发送到客户端,但它在启动时收到的第一个请求除外。当我第一次运行服务器程序(在LAN中的另一台计算机上)时,它会创建侦听套接字,绑定它,然后等待连接。在接收到第一个请求时,它将日期/时间字符串打印到它自己的标准输出(终端),而不是输出到套接字,并且客户端程序挂起,一直等待。但是,所有后续请求都会正确发送到客户端。使用gdb时,我注意到contfd总是设置为零。它在第一个请求时设置为零,在以后的所有请求中也设置为零。
我还有其他几个与此相关的问题:
如果服务器监听一个套接字(Listenfd),然后使用connect()重新连接另一个套接字(Connfd),客户端如何处理套接字的更改?我的理解是套接字由四个部分唯一标识:servIPaddr、servPort、clientIPaddr、clientPort
如何在没有root身份的情况下运行服务器(在Linux上)
如何干净利落地关闭侦听套接字,以便再次使用。如果我使用SIGINT(Ctrl-C)退出服务器程序,我会收到绑定错误。到目前为止,我一直在使用gdb,并使用"call close(Listenfd)"手动调用该函数。但是,如果我不使用gdb(即),有没有办法做到这一点呢?仅调试客户端应用程序)。
非常感谢您的帮助。
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <time.h>
#define BUFFER 80
int main(int argc, char **argv) {
int listenfd, connfd;
char buf[BUFFER];
struct sockaddr_in servaddr;
time_t ticks;
struct sockaddr *ptr;
char *ret;
if ( (listenfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
perror("socket error");
return 1;
}
memset(&servaddr, 0, sizeof(servaddr));
memset(buf, 0, BUFFER);
servaddr.sin_family = AF_INET;
servaddr.sin_port = htons(13);
servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
ptr = (struct sockaddr*) &servaddr;
if ( bind(listenfd, ptr ,sizeof(servaddr)) < 0) {
perror("bind error");
return 2;
}
if ( listen(listenfd, 128) < 0) {
perror("listen error");
return 3;
}
ptr = NULL;
while ( 1 ) {
if ( connfd = accept(listenfd, ptr, NULL) < 0) {
perror("accept error");
return 4;
} else {
ticks = time(NULL);
ret = ctime(&ticks);
sprintf(buf, "%.24s
", ret);
if ( write(connfd, buf, strlen(buf)) < 0) {
perror("write error");
close(connfd);
}
}
return 0;
}
推荐答案
我的预感是:在终端(Tty)上,标准输出和标准输入是相同的物理设备。因此,写入filedescriptor0(Stdin)实际上可能会起作用,并导致终端输出。
您需要用括号将其括起来
if ( connfd = accept(listenfd, ptr, NULL) < 0) {
喜欢这样
if ( (connfd = accept(listenfd, ptr, NULL)) < 0) {
或connfd
将分配"0"
更新刚才测试了这一点,这确实是罪魁祸首。下一次,使用gcc -Wall
编译,编译器会告诉您这一点(以及其他几个格式/风格良好的问题)。这样,您就不必依靠预感来查找错误。
已修复版本:
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <time.h>
#define BUFFER 80
int main(int argc, char **argv) {
int listenfd, connfd;
char buf[BUFFER];
struct sockaddr_in servaddr;
time_t ticks;
struct sockaddr *ptr;
char *ret;
listenfd = socket(AF_INET, SOCK_STREAM, 0);
if ( listenfd < 0 ) {
perror("socket error");
return 1;
}
memset(&servaddr, 0, sizeof(servaddr));
memset(buf, 0, BUFFER);
servaddr.sin_family = AF_INET;
servaddr.sin_port = htons(13);
servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
ptr = (struct sockaddr*) &servaddr;
if ( bind(listenfd, ptr ,sizeof(servaddr) ) < 0) {
perror("bind error");
return 2;
}
if ( listen(listenfd, 128) < 0 ) {
perror("listen error");
return 3;
}
ptr = NULL;
while ( 1 ) {
connfd = accept(listenfd, ptr, NULL);
if ( connfd < 0 ) {
perror("accept error");
return 4;
} else {
ticks = time(NULL);
ret = ctime(&ticks);
sprintf(buf, "%.24s
", ret);
if ( write(connfd, buf, strlen(buf)) < 0) {
perror("write error");
close(connfd);
}
}
}
return 0;
}
这篇关于服务器打印到标准输出,而不是套接字的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!