21.1 进程查看和进程状态
21.1.1 ps 指令
ps 指令通常可以查看到进程的 ID、进程的用户 ID、进程状态和进程的 Command
ps:查看当前用户启动的进程
ps -ef:详细查看后台进程信息,可以用 ps -ef | more 进行分屏查看
ps -aux:当前运行的进程占用的CPU的时间,内存的时间
- USER:进程的属组
- PID:进程的ID
- PPID:父进程
- %CPU:进程占用的CPU百分比
- %MEM:占用内存的百分比
- NI:进程的 NICE值,数值大,表示较少占用CPU的时间
- VSZ:进程虚拟大小
- RSS:驻留中页的数量
- TTY:终端ID
- WCHAN:正在等待的时间
- START:启动进程的时间
- STAT:进程的状态
- TIME:进程消耗CPU的时间
- COMMAND:命令的名称和参数
21.1.2 进程状态
- linux上进程有5种状态:
- 运行(正在运行或在运行队列中等待)
- 中断(休眠中, 受阻, 在等待某个条件的形成或接受到信号) ---可中断等待状态
- 不可中断(收到信号不唤醒和不可运行, 进程必须等待直到有中断发生)---不可中断等待状态
- 僵死(进程已终止, 但进程描述符存在, 直到父进程调用wait4()系统调用后释放)
- 停止(进程收到SIGSTOP, SIGSTP, SIGTIN, SIGTOU信号后停止运行运行)
- ps工具标识进程的5种状态码:
- D 不可中断 uninterruptible sleep (usually IO)
- R 运行 runnable (on run queue)
- S 中断 sleeping
- T 停止 traced or stopped
- Z 僵死 a defunct ("zombie") process
注: 其它状态还包括W(无驻留页), <(高优先级进程), N(低优先级进程), L(内存锁页).
21.2 进程调度和进程状态变化
21.2.1 进程调度
进程如何进行调度:
- 第一步:处理内核中的工作
- 第二步:处理当前进程
- 第三步:选择进程
- 实时进程
- 普通进程
- 第四步:进程交换
启动进程之后,会启动一个 task_struct 结构体,在此结构体中会存储进程调度的一些信息。task_struct 中的调度信息:
- 策略
- 轮流策略
- 先进先出策略
- 优先权
- Jiffies 变量
- 实时优先权
- 实时进程之间
- 计数器
21.2.2 进程状态变化关系
21.3 进程标识
内核通过 PID 来区分不同的进程
#include <unistd.h>
#include <sys/types.h> pid_t getpid(void); //获得当前进程ID
uid_t getuid(void);//获得当前进程的实际用户ID
uid_t geteuid(void);//获得当前进程的有效用户ID
gid_t getgid(void);//获得当前进程的用户组ID
pid_t getppid(void);//获得当前进程的父进程ID
pid_t getpgrp(void);//获得当前进程所在的进程组ID
pid_t getpgid(pid_t pid);//获得进程ID 为 pid 的进程所在的进程组 ID
- 实际用户:即登陆的用户ID,自己的账号登陆系统后,输入 id ,即可查看到自己的用户ID,当前登陆进去的用户即为实际用户。可以用当前用户启动很多进程。
- 有效用户:在登陆实际用户后,我们仍然可以用不同的用户来启动进程,比如 root 用户。这种就使在执行进程的时候以一种特定的身份去启动,这个身份就是有效用户。一般情况下,有效用户就是实际用户。
例子:process_id.c
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h> int main(void)
{
printf("pid: %d\n", getpid());//获得当前启动进程的进程编号
printf("uid: %d\n", getuid());//获得当前实际用户的ID
printf("ppid: %d\n", getppid());//获得当前执行进程的父进程号
printf("euid: %d\n", geteuid());//获得当前有效用户的 ID
printf("user gid: %d\n", getgid());//获得用户的组ID
printf("gid: %d\n", getpgrp());//获得当前进程组的ID
printf("pgid: %d\n", getpgid(getpid()));//获得指定进程的进程组ID
printf("ppgid: %d\n", getpgid(getppid()));//获得当前父进程所在的进程组的ID return ;
}
当前 ppid 即父进程ID为8468,为当前 shell 的进程
uid 为用户组 ID,我们当前用户的id 为1000,用户组的 ID 为 1000
实际用户和有效用户区别:
查看当前要运行的进程的所在组,可以看见文件的实际用户和实际用户组都为 rk3399
进入 root 模式修改文件的所属组:sudo chown root.root bin/process_id
修改粘着位,通过修改粘着位可以修改启动进程的有效用户:
sudo chmod u+s bin/process_id //增加粘着位
s 就是粘着位的设置,为文件添加有效用户
可以看见增加了一个 s,执行权限变为了 s
运行程序:
可以看见实际用户还是我们当前的用户,有效用户 euid 变为了 root
当针对文件拥有者本身修改粘着位以后,在运行进程的时候,可以将原先用户的权限(rk3399)提升位文件拥有者(root)的权限。