进程控制

1.进程的创建主要由fork和vfork函数实现:

vfork和fork都是创建子进程的函数;vfork创建的子进程和父进程共享内存空间,fork创建的子进程和父进程具有独立的地址空间;fork创建的子进程和父进程的执行顺序取决于操作系统调度,而vfork创建的子进程一定先于父进程,在它调用exec(exit)之后父进程才可能被进程调用;

int main()
{
  int ret=fork();
  printf("proc:%d,ret:%d\n",getpid(),ret);
  sleep(1);
  return 0;

}

int var=10;
int main()
{
  printf("var=%d\n",var);
  pid_t pid;
  pid=vfork();
  if(pid==-1)
  {
    perror("vfork()");
    exit(1);
  }
  if(pid==0)
  {
    sleep(5);
    var=100;
    printf("%d\n",var);
    exit(0);

  }

2.进程的等待由wait函数实现:

int main()
{
  pid_t pid;
  pid=fork();
  if(pid==-1)
  {
    perror("fork()");
    exit(1);
  }
  else if(pid==0)
  {
    sleep(20);
    exit(10);
  }
  else 
  {
    int status;
    int ret=wait(&status);
    if(ret>0&&(status&0X7F)==0)
    {
      printf("child exit code:%d\n",(status>>8)&0XFF);
    }
    else if(ret>0)
    {
      printf("sig code:%d\n",status&0X7F);
    }
  }

}

3.进程的终止

exit和_exit的区别:
_exit()执行后立即返回给内核,而exit()要先执行一些清除操作,然后将控制权交给内核。
调用_exit函数时,其会关闭进程所有的文件描述符,清理内存以及其他一些内核清理函数,但不会刷新流(stdin, stdout, stderr  ...).   exit函数是在_exit函数之上的一个封装,其会调用_exit,并在调用之前先刷新流。

int main()
{
  printf("hello\n");
  printf("world\n");
  _exit(0);

}

int main()
{
  printf("hello\n");
  printf("world");
  exit(0);

}

4.popen,system,fork

(1)popen函数

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

如果调用 fork() 或 pipe() 失败,或者不能分配内存将返回NULL,否则返回标准 I/O 流。

popen 没有为内存分配失败设置 errno 值。
popen 没有为内存分配失败设置 errno 值。

如果调用


fork() 或 pipe() 时出现错误,errno 被设为相应的错误类型。

(2)system函数

system()函数先fork一个子进程,在这个子进程中调用/bin/sh -c来执行command指定的命令。/bin/sh在系统中一般是个软链接,指向dash或者bash等常用的shell,-c选项是告诉shell从字符串command中读取要执行的命令(shell将扩展command中的任何特殊字符)。父进程则调用waitpid()函数来为变成僵尸的子进程收尸,获得其结束状态,然后将这个结束状态返回给system()函数的调用者。

(3)fork函数

一个现有进程可以调用fork函数创建一个新进程。由fork创建的新进程被称为子进程(child process)。fork函数被调用一次但返回两次。两次返回的唯一区别是子进程中返回0值而父进程中返回子进程ID。
子进程是父进程的副本,它将获得父进程数据空间、堆、栈等资源的副本。注意,子进程持有的是上述存储空间的“副本”,这意味着父子进程间不共享这些存储空间。
5.

 vfork 时父子进程共享数据段,fork时进行拷贝,如果,vfork子进程中使用return返回,则会出现段错误  :

vfork后父子进程是共用一个栈的。内核会让父进程暂时挂起,等子进程退出后在执行(或者释放其所使用的地址空间,比如调用exec函数)。子进程从vfork中return,调用栈就给破坏了,等子进程退出,父进程在运行时,就会出错误了。   







  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值