使用 select 从套接字和标准输入读取

问题描述 投票:0回答:2

我正在编写一个基于 ncurses 的聊天程序。起初,我只编写了网络内容(没有 ncurses),一切都工作正常,但添加图形后我无法让客户端应用程序正常工作。

主要问题是同时从标准输入和套接字读取。在无 ncurses 的版本中,我使用了 pthread,它的工作方式就像魅力一样。唉,似乎 pthread 和 ncurses 不能很好地结合在一起,所以我不得不寻找另一个解决方案。 我认为 select() 可以,但它仍然只从 stdin 读取并完全忽略套接字。

这是完整代码:代码

有趣的部分是:

char message[1024];
fd_set master;
fd_set read_fds;

FD_ZERO(&master);
FD_ZERO(&read_fds);

FD_SET(0,&master);
FD_SET(s,&master); // s is a socket descriptor
while(true){
read_fds = master;
if (select(2,&read_fds,NULL,NULL,NULL) == -1){
  perror("select:");
  exit(1);
}
// if there are any data ready to read from the socket
if (FD_ISSET(s, &read_fds)){
  n = read(s,buf,max);
  buf[n]=0;
  if(n<0)
  {
    printf("Blad odczytu z gniazdka");
    exit(1);
  } 
  mvwprintw(output_window,1,1,"%s\n",buf);
}
// if there is something in stdin
if (FD_ISSET(0, &read_fds)){
  getstr(message);
  move(CURS_Y++,CURS_X);
  if (CURS_Y == LINES-2){
    CURS_Y = 1;
  }
  n = write(s,message,strlen(message));
  if (n < 0){
    perror("writeThread:");
    exit(1);
  }
}
}

我可能不完全理解 select() 是如何工作的,或者也许我不应该 connect() 套接字..我在这里迷失了。我将不胜感激任何帮助!谢谢。

c sockets stdin ncurses posix-select
2个回答
6
投票

您的问题出在

select()

第一个参数不是您传入read_fds的文件描述符的数量,而是最高的套接字ID + 1。

来自手册页:

在每组中检查第一个 nfds 描述符;即 检查描述符集中从 0 到 nfds-1 的描述符。 (例如:如果您设置了两个文件描述符“4”和“17”,则 nfds 不应为“2”,而应为“17 + 1”或“18”。)

因此,在您的代码中,尝试传递“s+1”,而不是“2”。


1
投票

您需要指定最高的文件描述符来选择:

if (select(s + 1,&read_fds,NULL,NULL,NULL) == -1){

select() 需要知道它应该监视的文件描述符的数量。

© www.soinside.com 2019 - 2024. All rights reserved.