1

我有以下程序,它基本上是从键盘读取字符(getch() 无需单击“ENTER”即可执行此操作,功能取自此处:Capture characters from standard input without waiting for enter to be press),然后如果char 是合法动作之一(左、右等),它将它写入到 draw.out 正在读取的管道中。

draw.out 正在读取它收到的输入并打印到屏幕上。(假设该程序运行良好,我保证问题仅在以下代码中)。

问题是:

  1. draw.out 的行为就像它一遍又一遍地接收我的输入。这意味着,出于某种原因,即使我只按了一次 DOWN 键,它也会发送 draw.out ,就好像我点击了很多次一样。

  2. 发送一个输入后,我无法再发送,就好像循环正在停止一样。

试图打破我的头脑好几个小时......我真的很感激帮助。

int main(int argc, const char* argv[])
{
pid_t child_pid;
int fda[2];
if(pipe(fda)<0)
    perror("pipe error");

if((child_pid=fork())<0)
    perror("fork error");
else
{
    //if we're in father
    if(child_pid>0)
    {
        char c=' ';

        //close read side
        close(fda[0]);

        while(c!=QUIT)
        {
            //get an instruction from keyboard
            while(c!=ROTATE && c!=LEFT && c!=RIGHT &&
                    c!=DOWN && c!=QUIT)
            {
                c=getch();
            }

            //write the instruction to pipe
            write(fda[1],&c,1);
            //notify the child
            kill(child_pid,SIGUSR2);

        }
    }
    //if we're in child process
    else
    {
        dup2(fda[0],0);
        close(fda[0]);
        close(fda[1]);
        execl("./draw.out","./draw.out",NULL);
    }
}

//close everything
close(fda[0]);
close(fda[1]);
return 0;
}

//this function works well in linux with tsch installed or in SSH bash shell
char getch()
{
char buf = 0;
struct termios old = {0};
if (tcgetattr(0, &old) < 0)
        perror("tcsetattr()");
old.c_lflag &= ~ICANON;
old.c_lflag &= ~ECHO;
old.c_cc[VMIN] = 1;
old.c_cc[VTIME] = 0;
if (tcsetattr(0, TCSANOW, &old) < 0)
        perror("tcsetattr ICANON");
if (read(0, &buf, 1) < 0)
        perror ("read()");
old.c_lflag |= ICANON;
old.c_lflag |= ECHO;
if (tcsetattr(0, TCSADRAIN, &old) < 0)
        perror ("tcsetattr ~ICANON");
return (buf);
}
4

2 回答 2

1

发送 char 后,您需要将其设置为,或c以外的值。ROTATELEFTRIGHTDOWN

如果你不这样做,getch()将永远不会被称为第二次......

所以在外部while(就在 之后kill)或循环开始(就在while包含getch()调用之前)的末尾添加类似

if (c != QUIT)
  c = ' ';
于 2013-04-20T15:03:56.267 回答
1

将 while 循环更改为 do/while 循环,以确保始终至少调用一次 getch()。

do
{
    c = getch();
}
while (c != ROTATE && c != LEFT && c != RIGHT && c != DOWN && c != QUIT);
于 2013-04-20T15:09:15.413 回答