【问题标题】:How to open new terminal through C program in linux如何在linux中通过C程序打开新终端
【发布时间】:2013-04-18 01:41:13
【问题描述】:

我已经编写了有很多连接的客户端-服务器代码,假设每个节点代表同一台机器上的不同进程。为此,我显然使用了 fork()。

但现在的问题是所有结果都显示在同一个终端上。 我想知道是否有任何方法可以在每个fork() 或进程创建后打开新终端并在特定终端上显示该进程的所有结果。

P.S:我尝试过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个命令左右?因为我不想在这部分编码上投入太多。

提前致谢!!!

【问题讨论】:

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

标签: c linux unix gnome-terminal


【解决方案1】:

也许你想要这样的东西。该程序使用 unix98 伪终端 (PTS),它是主从之间的双向通道。因此,对于您所做的每个分叉,您都需要通过在主端调用三元组 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)); 应该是 char * 的 %s。
  • 为什么i==0是父亲。通常PID==0 是子进程,对吧?您想写的更改i!=0
  • @humanityANDpeace from fork 手册页:成功时,子进程的 PID 在父进程中返回,在子进程中返回 0。 所以,是的,如果i==0你在子进程中。在这个例子中,正确的应该是i!=0。谢谢
猜你喜欢
  • 2020-07-23
  • 1970-01-01
  • 2023-03-30
  • 2020-04-08
  • 2016-08-16
  • 1970-01-01
  • 2015-05-07
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多