zoukankan      html  css  js  c++  java
  • c语言exit和return区别,在fork和vfork中使用

    c语言exitreturn区别,在forkvfork中使用

    exit函数在头文件stdlib.h中。

    exit0):正常运行程序并退出程序;

    exit1):非正常运行导致退出程序;

    return():返回函数,若在main主函数中,则会退出函数并返回一值,可以写为return0),或return 0

      1. return返回函数值,是关键字;exit是一个函数。

      2. return是语言级别的,它表示了调用堆栈的返回;而exit是系统调用级别的,它表示了一个进程的结束。

      3. return是函数的退出(返回)exit是进程的退出。

      4. returnC语言提供的,exit是操作系统提供的(或者函数库中给出的)。

      5. return用于结束一个函数的执行,将函数的执行信息传出个其他调用函数使用;exit函数是退出应用程序,删除进程使用的内存空间,并将应用程序的一个状态返回给OS,这个状态标识了应用程序的一些运行信息,这个信息和机器和操作系统有关,一般是 0 为正常退出,非0 为非正常退出。

      6. 非主函数中调用returnexit效果很明显,但是在main函数中调用returnexit的现象就很模糊,多数情况下现象都是一致的。

     

     

    exit(1)表示异常退出.这个1是返回给操作系统的不过在DOS好像不需要这个返回值   

    exit(0)表示正常退出   

     

    数字0,1,-1会被写入环境变量ERRORLEVEL,其它程序可以由此判断程序结束状态。   

    一般0为正常推出,其它数字为异常,其对应的错误可以自己指定。

     

    exit()就是退出,传入的参数是程序退出时的状态码,0表示正常退出,其他表示非正常退出,一般都用-1,标准C里有EXIT_SUCCESSEXIT_FAILURE两个宏,用exit(EXIT_SUCCESS);可读性比较好一点。
    					

     

     

     

     下面是几个例子:

    1.

    #include <unistd.h>

    #include <stdio.h>

    #include <stdlib.h>

    int main(void)

    {

    pid_t pid;

    int count=0;

    pid=vfork();

    if(pid==0)

    {

    printf("child: count=%d ",count);

    printf("child: getpid=%d ",getpid());

    count=1;

    printf("child: count=%d ",count);

    // return 0;

    exit(0); //ok

    }

    else

    {

    printf(" father: pid=%d ",pid);

    printf("father: count=%d ",count);

    }

    return 0;}

     运行结果

    [root@localhost part1_linux]# gcc fork2.c

    [root@localhost part1_linux]# ./a.out

    child: count=0

    child: getpid=9911

    child: count=1

    father: pid=9911

    father: count=1

    运行结果说明:vfrok时父、子进程共享数据段,fork时是进行拷贝。如果,vfork子进程中,使用return返回时,出现段错误,结果如下:

    [root@localhost part1_linux]# gcc fork2.c

    [root@localhost part1_linux]# ./a.out

    child: count=0

    child: getpid=10864

    child: count=1

    father: pid=10864

    father: count=0

    段错误

    2. 为什么执行结果子进程打印出来 我的父亲是id:1,与父进程id不同

    #include <stdio.h>

    #include <sys/types.h>

    #include <unistd.h>

    int main()

    {

        int i=0;

        pid_t pid;

        printf("还没创建子进程 ");

        i++;

        pid = fork();

        if(pid==-1)

        {

          printf("fork error! ");

        }

        else if(pid==0)

        {

          i++;

          printf("我是子进程,id%d ",getpid());

            printf("我的父亲是id:%d ",getppid());

             printf("-----i=%d----- ",i);

        }

        else

        {

          i++;

          printf("我是父进程,id:%d ",getpid());

          printf("-----i=%d----- ",i);

        }

        exit(0);

    }

    子进程在打印第一句时,父进程也在打印第一句,但是子进程在执行第二句时,父进程已经直接over了(这只是个简单的说法,实际过程可能并不如此,我要说的是,父进程先于子进程的打印语句之前就结束)。因此此时的子进程成了孤儿进程,会被init也就是1号进程领养,成为init的子进程。 为了避免这样的情况,父进程最后可以执行wait来等待子进程的返回。

    3. 用vfork()创建子进程,执行后程序一直不断地重复运行,不断创建子进程,结尾用exit(0)代替return(0)后问题就能解决

    return 0在一个函数中是正常的返回过程,它会使得程序返回到函数被调用处,回复之前的执行流程,return 语句不会被执行。而exit 一般是在任意位置和使用的,执行到exit 0时,整个进程就over了(这也是为什么在多线程程序中不要随意使用exit的原因),用来使程序退出运行,一般用来处理(不可挽回的)错误状态。

    #include <stdio.h>

    #include <sys/types.h>

    #include <unistd.h>

    int main()

    {

      int i=0;

      pid_t pid;

      printf("还没创建子进程 ");

      i++;

      pid = vfork();

      if(pid==-1)

      {

    printf("fork error! ");

      }

      else if(pid==0)

      {

    i++;

    printf("我是子进程,id%d ",getpid());

    printf("我的父亲是id:%d ",getppid());

    printf("-----i=%d----- ",i);

      }

      else

      {

    i++;

    printf("我是父进程,id:%d ",getpid());

    printf("-----i=%d----- ",i);

      }

      return(0);

    }

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

     

  • 相关阅读:
    ABP框架(asp.net core 2.X+Vue)运行前端(转)
    Spring学习指南第3版例子程序导入idea20.02
    Docker运行jar包(转)
    centos7使用docker搭建运行环境并部署jar(转)
    centos7修改docker镜像源的方法(转)
    虚拟机下CentOS7安装Docker(转)
    Virtual Box配置CentOS7网络(图文教程)(转)
    虚拟机docker运行jar程序
    docker安装
    【591】通过 Terminal 直接获取 GitHub 源码
  • 原文地址:https://www.cnblogs.com/timssd/p/4108853.html
Copyright © 2011-2022 走看看