【Linux】进程控制

一.
1.进程创建
fork函数:它从已存在进程中创建一个新进程。新进程为子进程,而原进程为父进程。

#include<unistd.h>
pid_t fork();
返回值:子进程返回0,父进程返回子进程id,出错返回-1

2.进程终止
常见退出方法
(1)从main返回:退出前刷新缓冲区
(2)调用exit:退出前刷新缓冲区,任意位置调用都是退出进程
(3)_exit:不刷新缓冲区,缓冲区数据被丢弃,任意位置调用都是退出进程

_exit函数

#include<unistd.h>
void _exit(int status);

exit函数

#include<unistd.h>
void exit(int status);

return退出
调用main的运行时函数会将main的返回值当作exit的参数

3.进程等待
获取退出子进程的返回值,避免子进程成为僵尸进程
wait方法

#include<sys/types.h>
#include<sys/wait.h>
pid_t wait(int* status);
返回值:成功返回被等待进程pid,失败返回-1
参数:输出型参数,获取子进程退出状态

waitpid方法

pid_t waitpid)pid_t pid,int* status,int options);
返回值:正常返回时waitpid返回收集到的子进程的进程ID
若设置了WNOHANG,而调用中waitpid发现没有已退出的子进程可收集,则返回0
调用中出错,返回-1,errno会被设置成相应的值以指示错误所在

二.编写自主shell

#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<string.h>
#include<sys/wait.h>
#include<ctype.h>
int main()
{
  while(1)
  {
    printf("[liyuchen@localhost]$");
    fflush(stdout); 
    char tmp[1024]={0};
    scanf("%[^\n]%*c",tmp);
    printf("[%s]\n",tmp);
    //需要将整体字符串解析出:程序名称+参数
    //"ls -a -l"->"ls" "-a" "-l"
    char *ptr=tmp;
    int argc=0;
    char *argv[32]={NULL};
    while(*ptr!='\0')
    {
      if(!isspace(*ptr))
      {
        //指针走到非空白字符处
        argv[argc]=ptr;
        argc++;
        while(!isspace(*ptr)&&*ptr!='0')ptr++;
        *ptr='\0';
        ptr++;
        continue;
      }
      ptr++;
    }
    argv[argc]=NULL;
    if(!strcmp(argv[0],"cd"))
        {
          //改变当前工作路径
          chdir(argv[1]);
          continue;
        }
    int pid=fork();
    if(pid==0)
    {
      execvp(argv[0],argv);
      //若子进程程序替换失败,则直接退出,因为终端不需要多个shell
      exit(0);
    }
    //等待子进程退出,避免僵尸进程
    wait(NULL);
  }
}

在这里插入图片描述
三.
1.popen
popen()函数通过创建一个管道,调用fork产生一个子进程,执行一个shell以运行命令来开启一个进程。这个进程必须由pclose()函数关闭,而不是fclose()函数。pclose()函数关闭标准I/O流,等待命令执行结束,然后返回shell的终止状态。如果shell不能被执行,则pclose()返回的终止状态与shell已执行exit一样。

2.system
system()会调用fork()产生子进程,由子进程来调用/bin/sh-c string来执行参数string字符串所代表的命令,此命令执行完后随即返回原调用的进程。在调用system()期间SIGCHLD 信号会被暂时搁置,SIGINT和SIGQUIT 信号则会被忽略。

3.popen/system和fork的区别
(1)执行流程
system:在执行期间调用进程会一直等待shell命令执行完成
popen:无须等待shell命令执行完成就返回(并行执行),popen后需要调用pclose防止子进程变成”僵尸”状态。
fork :执行期间父进程等待子进程的退出码

(2)信号处理
system:对SIGCHLD、SIGINT、SIGQUIT都做了处理,
popen: 没有对信号做任何的处理。

system()调用堆信号屏蔽的原因是因为system能够及时的退出
并且能够正确的获取子进程的退出状态(成功回收子进程)。
popen()函数中没有屏蔽SIGINT、SIGQUIT的原因
是因为popen是”并行的”,不能影响其它”并行”进程。

(3)功能
system:执行shell命令最后返回是否执行成功,
popen:执行命令并且通过管道和shell命令进行通信。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值