如何在Linux中通过C程序打开新的终端窗口

6
我编写了客户端-服务器代码,其中有许多连接,假设每个节点表示同一机器上的不同进程。为此,我显然使用了 fork()。
但现在问题是所有的结果都显示在同一个终端上。 我想知道是否有办法在每个 fork() 或进程创建后打开新终端,并将所有结果显示在特定终端上的方法。
附言:我尝试过 system("gnome-terminal"),但它只是打开新终端,所有的结果仍然只显示在同一个终端上。所有的新终端只是打开并保持空白,没有任何结果。
我也阅读了这个链接 How to invoke another terminal for output programmatically in C in Linux,但我不想用参数或其他方式运行我的程序。它应该像 ./test 一样简单。
下面是我的代码:-
for(int i=0;i<node-1;i++)
  {
    n_number++;
    usleep(5000);
    child_pid[i]=fork();
    if(!child_pid[i])
    {
      system("gnome-terminal");
      file_scan();
      connection();          
      exit(0);
    }     
    if(child_pid[i]<0)
      printf("Error Process %d cannot be created",i);
  }
  for(int i=0;i<node-1;i++)
    wait(&status);

基本上我想要的是,每个进程都应该有一个新的终端,只显示该进程的信息或结果。
我确切地想要:
- 在fork()之后,如果我有一些与进程1相关的数据,那么我希望它的输出在一个终端上显示。 - 每个进程都是如此。所以如果我有3个进程,就必须有3个终端,每个终端都必须只显示与进程相关的数据。
我知道可以使用IPC(进程间通信)来实现这一点,但是否有其他方法?我的意思是只需要2-3个命令之类的东西吗?因为我不想在编写这部分代码时投入太多时间。
提前谢谢!!!

原始进程如何安排写入终端?新终端中的哪个进程将显示您编写的信息? - Jonathan Leffler
@JonathanLeffler 我希望我能更好地记住文件描述符和流,但是你理论上不能打开一个新的终端并将 stdout 重定向到该终端存在的任何文件描述符吗?嗯,我想我可能误解了终端... - rliu
1
我的怀疑/期望是,在没有其他东西的情况下,终端会设置自己的 shell,并且使用自己的(惊喜)终端 (/dev/ttyXXX/dev/ptyXXX)。因此,需要进行一些仔细的管道设置,以便终端窗口的输入来自希望写入它的进程,并且您需要一个程序(也许是 cat?)来读取该输入并写入终端窗口。我没有做过这个,所以我不知道其中的技巧。(您尝试过 man gnome-terminal 吗?) - Jonathan Leffler
就像……在fork()后,如果我有一些与进程1相关的数据,那么我希望将其输出到一个终端,并且每个进程都是如此。因此,如果我有3个进程,则必须有3个终端,每个终端都必须显示与进程相关的数据。 - suraj_fale
@JonathanLeffler,是的,我查看了{man gnome-terminal},没有发现任何有趣的内容。我还尝试使用{system("xterm")}来实现这个目标,但到目前为止还没有成功。谢谢! - suraj_fale
1个回答

6
也许您需要这样的东西。该程序使用unix98伪终端(PTS),它是主从之间的双向通道。因此,对于每个fork,您将需要通过在主端调用三元组posix_openpt、grantpt、unlockpt以及在从端调用ptsname来创建一个新的PTS。不要忘记在每个侧面纠正初始文件描述符(stdin、stdout和sdterr)。
请注意,这只是一个证明概念的程序,因此我没有进行任何形式的错误检查。
#define _XOPEN_SOURCE 600 
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <libgen.h>
#include <string.h>
#include <fcntl.h>

int main() {
  pid_t i;
  char buf[10];
  int fds, fdm, status;

  fdm = posix_openpt(O_RDWR);
  grantpt(fdm);
  unlockpt(fdm);

  close(0);
  close(1);
  close(2);

  i = fork();
  if ( i != 0 ) { // father
    dup(fdm);
    dup(fdm);
    dup(fdm);
    printf("Where do I pop up?\n");
    sleep(2);
    printf("Where do I pop up - 2?\n");
    waitpid(i, &status, 0);
  } else {  // child
    fds = open(ptsname(fdm), O_RDWR);
    dup(fds);
    dup(fds);
    dup(fds);
    strcpy(buf, ptsname(fdm));
    sprintf(buf, "xterm -S%c/2", basename(buf));
    system(buf);
    exit(0);
  }
}

sprintf(buf, "xterm -S%c/2", basename(buf)); 应该使用 %s 来表示 char * - Patrick B.
为什么 i==0 是父进程?通常 PID==0 是子进程,对吧?你想要改成 i!=0 吗? - humanityANDpeace
来自fork手册的@humanityANDpeace:_成功时,子进程的PID将在父进程中返回,并且在子进程中返回0。_所以,是的,如果i==0,你就在子进程中了。在这个例子中,正确的应该是i!=0。谢谢。 - Amadeus

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接