繁体   English   中英

命令后如何保持我的外壳运行

[英]How to keep my shell running after a command

我在另一个Shell中运行此基本Shell程序。 我无法弄清楚为什么执行“ ls”后我的shell无法继续运行。 我没有出口,但可以回到原始外壳。 如果要使用它,我必须每次都运行我的shell程序。 我想这就是fork()应该做什么。 我只希望我的shell使用我用if else语句编码的exit命令退出。 任何建议将不胜感激。 哦,不要理会gettoks()解析器函数,我无法弄清楚如何使用它进行输入,因此我为字符串输入cmSTR编写了if else语句,而不是使用gettoks()解析器。 主要是因为我不知道如何将输入传递给它

#include <iostream>
#include <unistd.h>
#include <sys/types.h>
#include <errno.h>
#include <signal.h>
#include <cstdlib>
#include <sys/wait.h>

using namespace std;
// Initializing counters for trapping
static int cc_counter = 0;
static int cz_counter = 0;
static int cq_counter = 0;
//Functions for trap signal handler
void cc_handler( int signo )
{
++cc_counter;
}
void cz_handler( int signo )
{
++cz_counter;
}
void cq_handler( int signo )
{
++cq_counter;
}

//*********************************************************
//
// Extern Declarations
//
//*********************************************************
using namespace std;
extern "C"
{
extern char **gettoks();
} 


//*********************************************************
//
// Main Function
//
//*********************************************************
int main( int argc, char *argv[] )
{
// local variables
int ii;
char **toks;
int retval;

// initialize local variables
ii = 0;
toks = NULL;
retval = 0;
char buf[1000];//Initialize of size for current working directory
string cmSTR;//String to hold input
int status;//Initialization of status for fork()
pid_t pid;//Declaration of pid

// main (infinite) loop
while( true )
{
    signal( SIGINT, cc_handler );// Traps Ctrl+C
    signal( SIGTSTP, cz_handler);// Traps Ctrl+Z
    signal( SIGQUIT, cq_handler);// Traps Ctrl+\
  //prompt and show current working directory
  cout <<("RS_SHELL:") << getcwd(buf,1000) << "\t";
  getline(cin ,cmSTR);//read input from keyboard
  // if else loop to switch based on command input
  if(cmSTR == "ls")// if ls, then execute arguement
  {
    execl( "/bin/ls", "ls", NULL );//System call to execute ls

    }
    else if(cmSTR == "exit")//if exit, then execute block of code
    {
        cout << "Ctrl C entered: " << ++cc_counter << "times"<< endl;
        cout << "Ctrl Z entered: " << ++cz_counter << "times"<< endl;
        cout << "Ctrl Back Slash entered: " << ++cq_counter << "times"<< endl;
        exit(1);
        }
    else if(cmSTR == "guish")// if guish, execute guish shell
    {
        execvp("guish", NULL);
        }
        //if input is not any of previous commands then fork()
    else if(cmSTR != "ls" && cmSTR != "exit" && cmSTR != "guish" && cmSTR != "\n")
    {
        pid = fork();
        if (pid < 0)//Loop to fork parent and child process
        {
            fprintf(stderr, "Fork Failed");
            exit(-1);
            }
            else if (pid == 0)//Child process
        {
            execvp("guish", NULL);//system call to execute guish shell

            }
            else //Parent process
            {
                waitpid( -1, &status,0);
                exit(0);
                }
                }




  // get arguments
  toks = gettoks();

  if( toks[0] != NULL )
{
  // simple loop to echo all arguments
  for( ii=0; toks[ii] != NULL; ii++ )
    {
      cout << "Argument " << ii << ": " << toks[ii] << endl;
    }

  if( !strcmp( toks[0], "exit" ))
    break;
 }
    }

   // return to calling environment
   return( retval );
   }

如您所怀疑, execl及其相关功能将当前过程与新过程重叠。 因此,在启动lsexecl调用之后,您的程序将不再存在以保持运行。

如果要让shell程序在运行ls之后继续存在,则需要在调用execl( "/bin/ls", "ls", NULL ); 之前添加fork() execl( "/bin/ls", "ls", NULL );

同样,如果您希望ls的输出与shell出现在同一控制台中,正如我想的那样,您将需要将ls的输出通过管道传回到shell,然后将其输出到shell的控制台中。 请参阅编写我自己的外壳……卡在管道上? , 例如。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM