为什么UNIX使用init进程启动其他进程?
为什么UNIX使用init进程启动其他进程?
在UNIX系统中,当系统启动时,内核完成初始化后会启动第一个用户空间进程,通常是init
进程。init
进程负责启动和管理其他用户空间进程,而内核本身并不直接处理这些任务。为什么UNIX采用这样的设计,而不是让内核直接负责启动所有进程?本文将从多个角度分析这种设计的背后逻辑与优点。
0. UNIX是否真的使用init进程?
RTFS,通过观察Linux内核源代码(这里),可以发现这段代码:
if (!try_to_run_init_process("/sbin/init") ||
!try_to_run_init_process("/etc/init") ||
!try_to_run_init_process("/bin/init") ||
!try_to_run_init_process("/bin/sh"))
return 0;
panic("No working init found. Try passing init= option to kernel. "
"See Linux Documentation/admin-guide/init.rst for guidance.");
这说明Linux真的会尝试启动一个init进程,甚至如果启动失败就“罢工”了。
同时观察pstree
的输出:
VM: ~ > pstree
systemd─┬─ModemManager───2*[{ModemManager}]
├─acpid
├─2*[agetty]
├─dbus-daemon
├─mysqld───48*[{mysqld}]
├─nginx───2*[nginx]
├─sshd───sshd───sshd───bash───pstree
├─systemd-journal
├─systemd-logind
...
的确可以发现所有的进程都是由systemd“生长分化”(fork)出来的。
1. 为什么内核不直接启动所有进程?
尽管内核拥有启动进程的能力,但让内核直接启动所有进程并不符合UNIX的设计哲学。
职责分离
进程的生命周期管理(如创建、调度)由内核负责,而决定哪些进程需要启动及其配置由用户空间的init进程管理。这样可以让内核保持简洁,减少复杂性。灵活性
通过init
或其他替代方案,用户可以动态调整系统初始化逻辑,例如定义不同运行级别或自定义服务启动顺序(systemctl
)。而将这些逻辑交给内核则会限制这种灵活性。避免内核代码膨胀
如果内核直接负责启动所有用户进程,它将不得不处理大量复杂的逻辑,例如服务依赖、配置文件解析等。这会导致内核代码变得庞大且难以维护。
2. 职责分离与模块化设计
UNIX操作系统遵循职责分离的设计哲学,将不同的功能模块化处理:
init
负责: 高层逻辑,比如系统初始化和服务管理, 决定哪些服务或进程需要启动以及启动的顺序。- 内核负责: 提供底层机制(如
fork
、exec
)来实现进程创建和调度。
通过将进程管理的高层逻辑交由用户空间中的init
进程负责,内核可以保持专注于底层资源管理,简化了自身的设计和实现。
3. 灵活性和可配置性
init
进程是用户空间的程序,可以被系统管理员配置或替换为其他初始化系统(如systemd
或upstart
)。这种灵活性使得不同的UNIX系统可以根据需求自定义其初始化逻辑,而无需更改内核。
> ls /usr/sbin/init -l
lrwxrwxrwx 1 root root 20 Dec 17 04:23 /usr/sbin/init -> /lib/systemd/systemd
- 内核的任务: 启动第一个用户空间进程(PID为1的
init
)。 init
的任务: 配置系统、启动服务、管理运行级别。
这种设计避免了将系统初始化的逻辑写入内核,使得系统更易于维护和扩展。
4. 总结
UNIX选择通过init
进程启动和管理其他进程,而不是由内核直接完成,这是为了实现职责分离、增强灵活性、简化内核设计并提高系统的可维护性。
这种设计不仅遵循了UNIX的模块化哲学,还为系统的扩展性和灵活性提供了基础,体现了UNIX设计的简洁与优雅。
为什么UNIX使用init进程启动其他进程?的更多相关文章
- Chromium的GPU进程启动流程
转载请注明出处:http://www.cnblogs.com/fangkm/p/3960327.html 硬件渲染依赖计算机的GPU,GPU种类繁多,兼容这么多种类的硬件,稳定性是个大问题,虽然Chr ...
- Linux进程启动/指令执行方式研究
1. 通过glibc api执行系统指令 0x1:system() glibc api system是linux系统提供的函数调用之一,glibc也提供了对应的封装api. system函数的原型为: ...
- Android 跨进程启动Activity黑屏(白屏)的三种解决方案
原文链接:http://www.cnblogs.com/feidu/p/8057012.html 当Android跨进程启动Activity时,过程界面很黑屏(白屏)短暂时间(几百毫秒?).当然从桌面 ...
- Android系统init进程启动及init.rc全解析
转:https://blog.csdn.net/zhonglunshun/article/details/78615980 服务启动机制system/core/init/init.c文件main函数中 ...
- Linux---从start_kernel到init进程启动
“平安的祝福 + 原创作品转载请注明出处 + <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 ” ini ...
- 使用gdb跟踪Linux内核启动过程(从start_kernel到init进程启动)
本次实验过程如下: 1. 运行MenuOS系统 在实验楼的虚拟机环境里,打击打开shell,使用下面的命令 cd LinuxKernel/ qemu -kernel linux-/arch/x86/b ...
- 实验三:gdb跟踪调试内核从start_kernel到init进程启动
原创作品转载请注明出处<Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 如果我写的不好或者有误的地方请留言 ...
- Android系统启动流程(一)解析init进程启动过程
整体流程大致如下: 1.init简介 init进程是Android系统中用户空间的第一个进程,作为第一个进程,它被赋予了很多极其重要的工作职责,比如创建zygote(孵化器)和属性服务等.in ...
- Linux内核分析-使用gdb跟踪调试内核从start_kernel到init进程启动
姓名:江军 ID:fuchen1994 实验日期:2016.3.13 实验指导 使用实验楼的虚拟机打开shell cd LinuxKernel/ qemu -kernel linux-3.18.6/a ...
- ARM-Linux移植之(三)——init进程启动流程分析
我们通常使用Busybox来构建根文件系统的必要的应用程序.Busybox通过传入的参数来决定执行何种操作.当init进程启动时,实际上调用的是Busybox的init_main()函数,下面我们来分 ...
随机推荐
- 卡特兰数 Catalan 数列
卡特兰数 Catalan 数列 引入 有一个无限大的栈,进栈的顺序为 \(1,2,\cdots,n\),求有多少种不同的出栈序列. 设 \(h[n]\) 为 \(n\) 个数的出栈序列方案数. 可以这 ...
- vue-quill-editor 自定义图片上传和视频上传
安装 npm install vue-quill-editor --save 在main.js 引入 import 'quill/dist/quill.core.css' import 'quill ...
- 使用MySQL Shell 8.4.1-LTS 直接将数据复制到 MySQL实例
在之前的文章中,我谈到了如何使用 MySQL Shell 通过多线程过程来转储和加载数据,以及如何以不同格式导出表数据,然后可以将这些数据导入到新的 MySQL 实例中.这篇文章将讨论我们如何直接将数 ...
- MapStruct 入门
介绍(What?Why?How?) What? MapStruct is a code generator that greatly simplifies the implementation of ...
- 论文泛读《T-Miner: A Generative Approach to Defend Against Trojan Attacks on DNN-based Text Classification》
发表时间:2021 期刊会议:30th USENIX Security Symposium 论文单位:Virginia Tech 论文作者:Ahmadreza Azizi,Ibrahim Asadul ...
- ZCMU-1051
比较来说不太难其实,当然找到一定的公式这与前面的1033相识,都会用到f(i,j)=f(i-1,j)+f(i-1,j-1) 我们可以先从小部分看出来,一层可以整体或者两部分,在面对第i层看前面i-1层 ...
- ZCMU-1033
我觉得这位大佬说的已经很好了,可以直接看她的思路了: 大佬思路 但是她的代码没有考虑到1 1 1 1的情况, 代码思路 这个是可以的很长且没有注释: #include<bits/stdc++.h ...
- QueryFilter的子字段设定
QueryFilter有一个SubFields属性和一个AddField方法. 对于SubFields属性是这样描述的.The comma delimited list of field names ...
- regsvr32.exe使用
regsvr32.exe使用详解: regsvr32.exe是32位系统下使用的DLL注册和反注册工具,使用它必须通过命令行的方式使用,格式是: regsvr32 [/u] [/s] [/n] [ ...
- VTK 设置面片背面颜色
在上一篇文章切开了零件,发现零件内部和外部颜色一样,当需要不一样时,可以通过actor的SetBackfaceProperty方法设置背面属性. 代码跟上一篇几乎一样,只是给actor设置了SetBa ...