CentOS 8启动流程

一、bios与uefi

BIOS

Basic Input Output System的缩写,即“基本输入输出系统”,是计算机启动时首先加载的程序。它主要负责检测和设置硬件,并启动操作系统。BIOS于1975年首次出现,是一个行业标准的固件接口。

UEFI

Unified Extensible Firmware Interface的缩写,意为“统一可扩展固件接口”,是BIOS的替代方案。其前身是Intel在1998年开始开发的Inter Bot Initiative,后来演变为可扩展固件接口(EFI),并于2005年由统一可扩展固件接口论坛正式命名为UEFI。

UEFI的优势

支持更大硬盘容量:与传统BIOS+MBR只能支持2048G的硬盘分区和4个主分区不同,UEFI+GPT不受硬盘容量和分区数量限制。虽然Windows系统有限制,最多支持128个GPT磁盘分区,最大分区18EB,且GPT格式没有主分区和逻辑分区的概念。

容错特性:UEFI采用模块化构建,具有更强的容错和纠错能力。

鼠标操作:UEFI内置图形驱动,支持原生分辨率的图形环境,用户可以使用鼠标进行操作。

扩展性强:UEFI包含一个可编程的开放接口,厂商可以利用此接口扩展功能,如备份和诊断。

支持联网:无需进入操作系统,即可通过网络进行远程故障诊断。

二、CentOS8启动流程

现代计算机是软件与硬件的复杂组合,从接通电源到系统登录,需要软件和硬件的紧密配合。我们将探讨CentOS8 x86_64系统的启动过程,虚拟机的启动流程大致相同,但某些与硬件相关的步骤由虚拟机软件处理。

接通电源

系统固件(UEFI或BIOS)运行自检,并初始化部分硬件。

系统固件搜索可启动设备

启动设备可能是UEFI启动固件中配置的,也可能是按照BIOS中配置的顺序搜索所有磁盘上的主启动记录(MBR)。

读取启动加载器(boot loader)

系统固件从MBR中读取启动加载器,然后将控制权交给启动加载器。在CentOS8中,启动加载器为GRUB2。

grub.cfg

GRUB2从/boot/grub2/grub.cfg文件中加载配置并显示启动菜单,用户可选择要启动的内核。grub.cfg文件可以通过grub2-mkconfig命令结合/etc/grub.d/目录和/etc/default/grub文件生成。

initramfs

选择内核或到达超时时间后,启动加载器从磁盘加载内核(vmlinuz)和initramfs,并将其放入内存。initramfs包含启动时所需的硬件内核模块(驱动)和初始化脚本等。可以通过lsinitrd和dracut命令结合/etc/dracut.conf.d/目录查看和配置initramfs文件。

启动加载器放权

启动加载器将控制权交给内核。

内核初始化

内核在initramfs中寻找硬件驱动并初始化相关硬件,然后启动/usr/sbin/init(PID=1)。在CentOS8中,/sbin/init是systemd的链接。

启动initrd.target并挂载

systemd执行initrd.target包含的所有单元,并将根文件系统挂载到/sysroot/目录。initrd.target启动时的依赖单元将根据/etc/fstab设置挂载硬盘。

切换根文件系统

内核将根文件系统从initramfs切换到/sysroot(硬盘上的根文件系统),systemd会找到磁盘上安装的systemd并自动重新执行。

启动相应目标

硬盘上安装的systemd会查找从内核命令行传递的目标或系统中配置的默认目标,并启动相应单元,进入登录界面。默认目标是/etc/systemd/system/default.target。

注意:字符界面

如果默认目标为multi-user.target(字符界面),systemd会先执行sysinit.target初始化系统,然后执行basic.target与getty.target准备基本系统环境和终端,再启动multi-user.target下的相关应用,同时执行/etc/rc.d/rc.local(需要执行权限)和登录服务(systemd-logind.service),开启登录界面。

图形界面

豆包AI编程 豆包AI编程

豆包推出的AI编程助手

豆包AI编程 483 查看详情 豆包AI编程

在multi-user.target的基础上执行graphical.target启动所需的服务,开启图形界面。

三、运行级别切换与相关配置文件

CentOS8在启动过程中需要判断对应的运行级别,不同的运行级别会启动不同的单元。运行级别的切换涉及以下命令:

systemctl isolate …/init [0-6]:切换运行级别systemctl get-default:查看当前的默认运行级别systemctl set-default …:设置默认的运行级别runlevel:查看之前的运行级别和当前的运行级别

运行级别对照

runlevel0.target -> poweroff.targetrunlevel1.target -> rescue.targetrunlevel2.target -> multi-user.targetrunlevel3.target -> multi-user.targetrunlevel4.target -> multi-user.targetrunlevel5.target -> graphical.targetrunlevel6.target -> reboot.target

因此,当使用init [0-6]切换运行级别时,系统执行的是systemctl isolate runlevel[0-6].target。

四、Boot Loader与GRUB2

在CentOS8的启动流程中,有一个称为启动加载器(boot loader)的工具。如果没有这个boot loader,就无法加载内核。在CentOS8中使用的启动加载器是Grub2。在了解Grub2之前,先简单介绍一下boot loader。

Boot Loader的两个阶段

boot loader的工作过程分为两步:

stage1:执行boot loader的主程序,必须安装在MBR或启动扇区。由于空间限制,仅存放最小化的boot loader,不包含配置文件。stage2:通过主程序加载配置文件。boot loader的主程序加载所有相关的配置文件,包括定义文件系统类型的和CentOS8中的grub.cfg文件,这些文件通常存放在/boot目录中。

GRUB2

目前,Linux下的启动加载器有两种,一种是LILO,另一种是GRUB。由于GRUB功能更强大,支持的文件系统较多,因此越来越多的操作系统使用GRUB作为boot loader。CentOS从7开始使用功能更为强大的GRUB2。

GRUB2的优点

支持更多的文件系统开机时可以手动调整启动参数动态更新配置文件,修改完配置文件后不需要重新安装GRUB2

GRUB2与硬盘

由于grub2的主要任务是从硬盘中加载内核,所以grub2必须识别硬盘。grub2识别硬盘的方式与Linux系统有所不同。在Linux系统中,硬盘通常被识别为类似sda1的形式,而在grub2中,硬盘统一被识别为hd设备,排序方式全部用数字进行排序,而不是用字母加数字的混合形式。这样做的目的是为了定义grub2查找内核时的顺序。

grub2配置文件

grub2的配置文件建议不要随意更改。grub.cfg文件的内容由grub2-mkconfig命令自动生成,相关模板和设置存放在/etc/grub.d/目录以及/etc/default/grub文件中。如果需要修改,需要通过调整/etc/default/grub文件并使用grub2-mkconfig命令生成新的grub.cfg文件。

/etc/default/grub与/etc/grub.d/

grub2的配置文件grub.cfg内容复杂且不建议手动修改。如果需要修改,需要通过/etc/default/grub文件以及/etc/grub.d/目录内的内容来实现。

/etc/default/grub

GRUB_TIMEOUT=5:定义启动菜单默认等待时间,单位为秒。GRUB_DISTRIBUTOR=”$(sed ‘s, release .*$,,g’ /etc/system-release)”:定义获取操作系统名称的方式。GRUB_DEFAULT=saved:定义开机时默认启动的项目,可以是数字、标题名称或saved(表示默认启动上次成功启动的操作系统)。GRUB_DISABLE_SUBMENU=true:是否隐藏子菜单。GRUB_TERMINAL_OUTPUT=”console”:定义启动时的界面使用哪种终端输出,值包含console、serial、gfxterm、vga_text等。GRUB_CMDLINE_LINUX=”resume=/dev/mapper/cl-swap rd.lvm.lv=cl/root rd.lvm.lv=cl/swap rhgb quiet”:定义额外的启动参数。GRUB_DISABLE_RECOVERY=”true”:是否启用修复模式。GRUB_ENABLE_BLSCFG=true:是否启用bootloader规范。

修改完成后,需要使用grub2-mkconfig -o /boot/grub2/grub.cfg命令重新生成配置文件。

/etc/grub.d/

00_header:设置默认参数。00_tuned:额外调整的值。01_menu_auto_hide:与菜单隐藏相关的设置。01_users:与用户相关的设置。10_linux:与内核相关的设置。20_ppc_terminfo:与终端相关的设置。20_linux_xen:与虚拟化相关的设置。30_os-prober:与操作系统检测相关。30_uefi-firmware:与UEFI启动设置相关,需要硬件支持。40_custom&41_custom:自定义设置。

这些文件会按照数字顺序由小到大加载。

CentOS 8启动流程

注意:此为紧急模式,用于紧急处理系统错误,无法使用rescue.target时,可以尝试使用此模式。

以上就是CentOS 8启动流程的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/465366.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月8日 05:13:33
下一篇 2025年11月8日 05:18:10

相关推荐

  • 模板方法模式怎样工作 算法骨架与步骤重定义

    模板方法模式通过在抽象类中定义算法骨架并由子类实现具体步骤,实现流程固定、细节可变的设计;其核心是父类控制执行流程,子类提供差异化实现,确保代码复用与行为统一,常用于框架和标准化流程场景,最终完整实现了继承机制下的灵活扩展与结构稳定。 模板方法模式通过在一个抽象类中定义算法的骨架,将具体步骤的实现延…

    好文分享 2025年12月18日
    000
  • 怎样为C++配置高性能日志环境 spdlog库与异步日志系统搭建

    要配置c++++的高性能日志环境,应选用spdlog库并启用异步日志机制。1. spdlog基于fmt库,轻量且支持多种日志级别与多线程安全,具备异步日志功能;2. 启用异步日志需包含头文件、创建文件sink、构建异步logger并设置为全局默认,最后调用spdlog::shutdown()确保日志…

    2025年12月18日 好文分享
    000
  • 如何用C++处理日志文件滚动 按大小或时间分割日志方案

    c++++程序中可通过编程实现日志滚动。按大小分割:监控文件大小,超限后重命名并新建文件,如超过10mb则生成带时间戳的新文件;按时间分割:记录写入时间,超指定间隔(如24小时)创建新文件,每天一个日志便于归档;组合策略:每天基础文件下再按大小切分,如app_20250405_1.log等;注意事项…

    2025年12月18日 好文分享
    000
  • C++中堆和栈内存有什么区别 解释两种内存区域的特性和使用场景

    c++++中堆和栈的核心区别在于管理方式、生命周期、分配速度和使用场景。栈内存由系统自动管理,分配释放快,适用于小型局部变量和函数调用,生命周期随作用域结束而终止;堆内存需手动管理,灵活性高,适用于动态数据结构和跨函数对象,但存在内存泄漏和野指针风险。选择栈的场景包括:1. 小型固定大小的数据;2.…

    2025年12月18日 好文分享
    000
  • C++多核CPU如何避免伪共享 缓存行填充与对齐技术实践

    伪共享会导致多线程性能退化,解决方法是缓存行填充与对齐。伪共享是指多个线程修改各自独立的变量时,因这些变量位于同一缓存行而引发缓存频繁失效;识别方法包括使用perf、valgrind、intel vtune等#%#$#%@%@%$#%$#%#%#$%@_20dc++e2c6fa909a5cd6252…

    2025年12月18日 好文分享
    000
  • 怎样在C++中实现异常重抛 throw不带表达式的使用技巧

    在c++++中,throw;用于重新抛出当前捕获的异常,避免复制对象并保留其动态类型和上下文信息。1. throw;的基本作用是将catch块中捕获的异常原样抛出,保持异常对象的原始类型;2. 相比throw e;,它避免了对象切片、性能损耗及上下文信息丢失;3. 常见场景包括日志记录后重抛和资源清…

    2025年12月18日 好文分享
    000
  • 形式化验证:如何用SAT验证C++算法正确性

    形式化验证,简单来说,就是用数学的方法证明你的C++算法是不是真的像你想象的那样工作。SAT求解器在这里扮演了关键角色,它能帮你检查算法在所有可能输入下的行为,而不仅仅是靠几个测试用例。 用SAT验证C++算法正确性,本质上就是把C++代码转换成一个巨大的布尔表达式,然后用SAT求解器来判断这个表达…

    2025年12月18日 好文分享
    000
  • 类模板如何声明和实例化 模板类开发指南

    类模板的声明使用 template 或 template 语法,实例化需指定具体类型如 mytemplate,核心是通过泛型实现代码复用;1. 类模板声明以 template 开始,包含类型参数(typename 或 class)或非类型参数,如 template class mytemplate …

    2025年12月18日
    000
  • C++内存模型如何处理弱内存架构 ARM/PowerPC平台的差异

    c++++内存模型通过提供std::atomic和内存序(memory_order)语义来处理arm或powerpc这类弱内存架构的并发问题。1. 它允许开发者明确指定操作的可见性和顺序性要求,从而在不同平台上保持一致的行为;2. 通过封装底层硬件屏障指令,如arm的dmb或powerpc的sync…

    2025年12月18日 好文分享
    000
  • C++如何实现模板递归 C++模板递归技巧详解

    c++++模板递归是一种在编译期通过模板定义调用自身实现递归效果的元编程技术。其核心在于模板特化,通用模板处理一般情况,特化模板作为终止条件,如计算阶乘时通过factorial递归调用factorial并以factorial终止递归。模板递归的实际应用包括:1. 编译期计算(如阶乘、数组长度);2.…

    2025年12月18日 好文分享
    000
  • 如何解决C++模板编译错误?常见问题分析与修复方法

    c++++模板编译错误常见原因及解决方法如下:1. 声明与定义分离导致错误,应将模板声明和定义放在同一头文件中;2. “未定义的引用”问题可通过显式或隐式实例化模板解决;3. 类型不匹配可使用static_assert、std::enable_if或c++20 concepts进行类型约束;4. 模…

    2025年12月18日 好文分享
    000
  • C++多线程编程如何避免虚假共享 填充和内存对齐技术解析

    虚假共享是多线程编程中因不同变量共处同一缓存行导致的性能问题。1.它发生在多个线程修改位于同一缓存行的不同变量时,引发频繁缓存失效;2.填充可通过插入多余字节使变量分布于不同缓存行,如定义占满64字节的结构体;3.内存对齐用alignas确保变量按缓存行大小对齐,避免紧凑排列;4.结合std::ha…

    2025年12月18日 好文分享
    000
  • 如何在C++中实现RPC框架_远程调用原理详解

    实现c++++的rpc框架需从idl、序列化、网络传输等关键步骤入手。1. 使用protocol buffers或thrift作为idl定义服务接口和数据结构,并生成c++代码;2. 利用idl工具自动生成序列化与反序列化代码,用于数据转换;3. 选用boost.asio、grpc或zeromq等网…

    2025年12月18日 好文分享
    000
  • 怎样为C++配置静态链接环境 完全静态编译的可执行文件生成

    为c++++配置静态链接环境需安装支持静态链接的编译器如mingw-w64,下载或编译第三方库的静态版本,配置编译器链接选项使用-static、-static-libgcc、-static-libstdc++等标志,并指定静态库路径和手动解决依赖关系,最后测试可执行文件是否能独立运行;针对体积问题可…

    2025年12月18日 好文分享
    000
  • 什么时候使用C++的Pimpl惯用法 降低编译依赖与二进制兼容性

    pimpl惯用法通过将类的实现细节封装到一个私有指针指向的impl类中,显著减少编译依赖并保障二进制兼容性。1. 它将私有成员和实现细节移至源文件中,使头文件仅保留接口和前置声明,避免因实现变更引发大规模重编译;2. 由于类的大小和布局固定为指针大小,impl的变化不影响外部代码,确保库升级时abi…

    2025年12月18日 好文分享
    000
  • 如何配置C++的实时操作系统环境 QNX Momentic工具链设置

    要配置qnx momentic++工具链进行c++实时操作系统开发,需完成以下步骤:1. 从官方获取qnx sdp安装包并注册登录;2. 安装时选择c/c++开发工具、目标系统支持、momentics ide及文档示例等组件;3. 设置环境变量qnx_target、qnx_host和path以确保命…

    2025年12月18日 好文分享
    000
  • C++中数组越界访问有什么后果 未定义行为与内存错误分析

    数组越界访问是指访问数组元素时索引超出其合法范围,c++++不进行边界检查,导致未定义行为。例如int arr[5]访问arr[5]或arr[-1]即为越界。常见后果包括:1. 未定义行为使程序可能正常运行或崩溃,表现不确定;2. 写操作可能覆盖内存内容,引发段错误或逻辑异常;3. 破坏栈结构可能导…

    2025年12月18日 好文分享
    000
  • C++异常处理与SEH有何区别 Windows结构化异常处理对比

    c++++异常处理和windows seh的主要区别在于机制、使用场景和实现方式。1. c++异常是语言级机制,适用于逻辑错误处理,依赖try/catch/throw,跨平台兼容;2. seh是系统级机制,用于底层异常如硬件错误,依赖windows api,语法为__try/__except;3. …

    2025年12月18日 好文分享
    000
  • C++中如何定义和初始化数组?详解数组的基本语法

    c++++中定义和初始化数组的方法有多种,1. 定义时直接初始化,如int numbers[5] = {1,2,3,4,5};2. 初始化列表元素不足时,剩余元素自动初始化为0;3. 使用空初始化列表将所有元素初始化为0;4. c++11支持统一初始化语法;5. 动态数组使用new和delete手动…

    2025年12月18日 好文分享
    000
  • C++严格类型别名规则例外情况 使用std::bit_cast进行安全类型转换

    std::bit_c++ast是一种安全的类型转换方式,允许绕过c++严格类型别名规则以重新解释对象的位模式。其核心在于编译器基于类型别名规则进行优化时假设不同类型的指针不会指向同一内存区域,而std::bit_cast通过直接复制源对象的位模式并将其解释为目标类型来实现安全转换。1. 它与rein…

    2025年12月18日 好文分享
    000

发表回复

登录后才能评论
关注微信