任何计算机内部都包含一个基本的程序集合,成为操作系统(OS)。
操作系统包括:
对下,与硬件交互,管理所有的软硬件资源
对上,为用户程序(应用层程序)提供一个良好的执行环境

操作系统“管理”硬件。
如何管理:
在开发角度,操作系统对外会表现为一个整体,但是会暴露自身的部分接口,供上层开发使用,这部分由操作系统提供的接口,称为系统调用。
系统调用在使用上,功能比较基础,对用户的要求相对较高,所以,有心的开发者会对部分系统调用做封装,从而形成库,有了库,就很方便上层用户和开发者进行二次开发。
概念:进程是指程序的一个执行实例,正在执行的程序等。 内核层面概念:每个进程拥有独立的内存空间,系统资源和执行状态,是操作系统进行资源分配和调度和核心对象。
操作系统描述进程的的数据结构—PCB(process control block)
PCB基本概念:进程信息被放在一个叫做进程控制块的数据结构中,可以理解为进程属性的集合。

进程=内核数据结构+自己的代码和数据。
进程=task_struct+自己的代码和数据。
同一时刻,可能会有多个可执行程序被加载到内存,而操作系统要对这些可执行程序进行管理, 在操作系统层面上,就会在内部会建立一个task_stuct 对象,里面保存了该可执行程序的代码地址,数据地址等各种信息。将这些task_struct使用链表或者其他数据结构管理起来,那么操作系统对进程的管理就转化为对链表的管理。
1,进程的信息可以通过/proc系统文件夹查看
如:要获取PID为1的进程信息,你需要查看 /proc/1 这个文件夹。

2,大多数进程信息同样可以使用top和ps这些用户及工具来获取
ps aux #显示所有进程信息 top #动态查看进程资源占用
3,通过系统调用获取进程标识符
系统调用
getpid() #当前进程ID(PID) getppid() #父进程ID(PPID)


创建一个子进程,将父进程的task_struct会拷贝一份给子进程,但是会有部分的数据时需要修改的,比如进程的唯一标识符pid。子进程会共享父进程的代码数据,相当于发生了浅拷贝。
pid_t pid = fork(); if (pid == 0) { // 子进程代码 } else { // 父进程代码 }
操作系统中会存在多个进程。每个进程可能会存在不同的状态,有的进程正在运行,有的进程正在被调度,有的进程处于挂起等等。CPU要执行这些进程,这些进程的task_struct中会保存指向代码和数据的指针。操作系统会维护一个运行队列(running queue),存放要运行的task_struct,这个队列就叫做调度队列,其中的一个个task_struct就是要被调度的进程。

本质上,进程状态就是task_struct结构体中的一个变量,用来记录当前进程的状态,在操作系统内部通过宏定义的方式,标识不同的状态。
例如,下面的状态在Kernel源代码的定义:
static const char *const task_state_array[] = {"R (running)", /*0 */"S (sleeping)", /*1 */"D (disk sleep)", /*2 */"T (stopped)", /*4 */"t (tracing stop)", /*8 */"X (dead)", /*16 */"Z (zombie)", /*32 */};
只要进程在调度队列中,都处于运行状态。
以scanf为例,当执行scanf从键盘读取数据时,未输入数据时,操作系统会将当前进程的task_struct从调度队列中移除,放入设备的阻塞队列中,此时,该进程就处于阻塞状态。直到键盘响应,会将该进程重新放入调度队列的尾部,继续运行。
在磁盘上,会存在一块特定的分区,叫做swap分区。当内存资源严重不足时,操作系统会将一些不会被调度的进程的代码和数据交换到磁盘的swap分区,在操作系统内部只保留task_struct部分。此时的状态称为挂起状态。

阻塞挂起状态:将阻塞队列中的进程 唤出到磁盘的swap分区上
阻塞运行状态: 将运行队列末端的进程唤出到磁盘的swap分区上
我们创建子进程的目的,是为了让子进程完成某种工作的。子进程完成后,父进程需要直到完成的信息。所以子进程在执行完后,必须要等待父进程获取它的退出信息,获取之前,该子进程的状态就是Z状态,僵尸状态。
当一个进程结束后,它的代码和数据就会被释放,所以它的退出信息只会保存在task_struct中。
维护退出状态本⾝就是要⽤数据维护,也属于进程基本信息,所以保存在task_struct(PCB)中,
换句话说,Z状态⼀直不退出,PCB⼀直都要维护。
那⼀个父进程创建了很多⼦进程,就是不回收,就会造成内存资源的浪费。
因为数 据结构对象本⾝就要占用内存。
父进程如果提前退出,那么⼦进程后退出,进入Z状态之后,那该如何处理呢?
父进程先退出,子进程就称之为“孤儿进程”。
孤儿进程被1号进程领养,就是要被操作系统领养。此时的子进程会成为后台进程。