服务器打印到stdout而不是socket

xst*_*xst 3 c sockets network-programming client-server

我正在研究UNIX网络编程中的示例,我在这里将"daytimeclientserv.c"改编为此代码.服务器按预期将日期/时间字符串发送到客户端,但启动时收到的第一个请求除外.当我第一次运行服务器程序(在LAN中的另一台计算机上)时,它会创建侦听套接字,绑定它然后等待连接.收到第一个请求后,它会将日期/时间字符串打印到其自己的标准输出(终端)而不是套接字,并且客户端程序将永远等待.但是,所有后续请求都会正确发送给客户端.使用gdb,我注意到connfd总是设置为零.它在第一个请求以及所有未来的请求上都设置为零.

我还有一些与此相关的问题:

  • 如果服务器侦听一个套接字(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\n", ret);
      if ( write(connfd, buf, strlen(buf)) < 0) {
        perror("write error");
        close(connfd);
    }
  }

  return 0;
}
Run Code Online (Sandbox Code Playgroud)

seh*_*ehe 5

这是我的预感:在终端(tty)上,stdout和stdin是相同的物理设备.因此,写入filedescriptor 0(stdin)实际上可能会起作用并导致终端输出.

你需要围绕这个括号

   if ( connfd = accept(listenfd, ptr, NULL) < 0) {
Run Code Online (Sandbox Code Playgroud)

像这样

  if ( (connfd = accept(listenfd, ptr, NULL)) < 0) {
Run Code Online (Sandbox Code Playgroud)

或者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\n", ret);
            if ( write(connfd, buf, strlen(buf)) < 0) {
                perror("write error");
                close(connfd);
            }
        }
    }

    return 0;
}
Run Code Online (Sandbox Code Playgroud)