我有以下程序,它基本上是从键盘读取字符(getch() 无需单击“ENTER”即可执行此操作,功能取自此处:Capture characters from standard input without waiting for enter to be press),然后如果char 是合法动作之一(左、右等),它将它写入到 draw.out 正在读取的管道中。
draw.out 正在读取它收到的输入并打印到屏幕上。(假设该程序运行良好,我保证问题仅在以下代码中)。
问题是:
draw.out 的行为就像它一遍又一遍地接收我的输入。这意味着,出于某种原因,即使我只按了一次 DOWN 键,它也会发送 draw.out ,就好像我点击了很多次一样。
发送一个输入后,我无法再发送,就好像循环正在停止一样。
试图打破我的头脑好几个小时......我真的很感激帮助。
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);
}