linux学习笔记之进程

linux中,每个进程在创建时都会分配一个数据结构,称为进程控制块(PCB).用它来记录进程的外部特征,主要包含进程标识符、进程的当前状态、进程相应的程序和数据地址、进程的优先级 etc.

fork和vfork(创建进程)
fork系统调用的作用是复制一个进程,fork创建的进程称为子进程,原来的进程称之为父进程,子进程从父进程得到了数据段和堆栈段的复制,这些需要重新分配内存。fork之后的进程执行顺序是不确定的,取决于内核的调度算法。
fork调用的奇妙在于它仅调用一次却返回两次。父进程中返回子进程的进程ID,子进程中返回0,出现错误则返回负值。

fork出错可能原因:
1.进程数达到系统规定上限
2.系统内存不足(少见,系统没可分配的内存,处于崩溃的边缘.)

vfork的返回与fork相同。但是vfork不需要完全复制父进程的数据段,在子进程没调用exec或exit之前,子进程和父进程共享数据段。fork不对父子进程的执行次序进行限制,但在vfork调用中,子进程先运行,父进程挂起,直到调用exec或exit后执行次序才不会限制。

vfork创建的并不是真正意义上的进程,而是线程,因为它并没有独立的内存资源,子进程和父进程共享数据段。

#include<sys/types.h>
#include<unistd.h>
#include<stdio.h>
#include<stdlib.h>

int main()
{
    int count = 1;
    pid_t pid;
    printf("before creat son process");    //创建进程之前
    if( ( pid = vfork()) < 0){
        printf("error in fork!");
        exit(1);
    }
    else if(pid == 0){
        int i;
        for(i = 0;i < 10;i++){
            print("this is son process");
        }
        exit(1);            //退出子进程
    }
    else if(pid > 0){        //父进程
        printf("this is father,his pid is:%d and the child pid is:%d",getpid(),pid);
    }
    return 0;
}

exec函数族
int execve(const char *pathname,char *const argv[],char *const envp[]);

execve第1个参数path是被执行应用程序的完整路径,第2个参数argv就是传给被执行应用程序的命令行参数,第3个参数envp是传给被执行应用程序的环境变量。

成功则无返回值,出错返回-1。exec函数族的作用是根据指定的文件名找到可执行文件,并用它来取代调用进程的内容,换句话说,就是在调用进程内部执行一个可执行文件。这里的可执行文件既可以是二进制文件,也可以是任何Linux下可执行的脚本文件。

exit和_exit函数
终止一个进程。exit和_exit的区别在于exit函数调用之前会检查文件的打开情况,把文件缓冲区的内容写回文件;_exit则是直接使进程停止运行,清除内存空间,并销毁其在内核的各种数据结构。

wait和waitpid函数
pid_t wait(int *status);
pid_t waitpid(pid_t pid,int *status,int option);
成功返回进程ID,出错返回-1

进程一旦调用了wait,就立即阻塞自己,由wait自动分析当前进程的某个子进程是否已经退出,如果让他找到这样的一个已经变成僵尸的子进程wait就会收集子进程的信息,把他彻底销毁;如果没有找到就会一直阻塞,知道有一个出现。

僵尸进程:一个进程调用exit之后并没有马上消失,他已经放弃了内存空间,没有可执行代码,也不能被调度,仅仅在进程列表中保留一个位置,记载该进程的退出状态等信息

waitpid多出了两个可供用户控制的参数pid和options,可以用来等待指定的进程,可以使进程不挂起立刻返回。

pid>0 只等待进程ID为pid的子进程,只要指定的子进程没结束,waitpid就会一直等待
pid=-1 等待任何的子进程退出 等价于wait
pid=0 等待同一进程组的任何子进程
pid<-1 等待一个指定进程组中的任何子进程,进程组的ID等于pid的绝对值

options一般有两个选项
WNOHANG 即使没有子进程退出,它也会立即返回,不会永远等待下去。如果设置了此选项,waitpid返回0
WUNTRCED常量意思是若实现支持作业控制,而与pid指定的任一子进程已处于暂停状态,并且其状态自暂停以后还没报告过,则返回其状态

#include<sys/types.h>
#include<sys/wait.h>
#include<unistd.h>
#include<stdio.h>
#include<stdlib.h>

int main()
{
    pid_t pc,pr;
    if( ( pc = fork()) == -1){
        printf("failed to create a new process\n");
        exit(0);
    }
    else if(pc == 0){
        sleep(10);       //子进程休眠10s
        exit(0);
    }
    do{
        pr = waitpid(pc,NULL,WNOHANG); //waitpid不会在这等待
        if(pr == 0){           //没有收集到子进程
            printf("No child exited\n");
            sleep(1);
        }
    }while(pr == 0);
    if(pr == pc){
        printf("successfully get child %d\n",pr);
    }
    else{
        printf("some error occured\n");
    }
    return 0;
}

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值