macos启动项真实加载顺序由launchd依赖关系、plist键值及目录扫描阶段决定:先系统级launchdaemons,再各级launchagents,最后smloginitem登录项。

如果您希望了解 macOS 系统中启动项的实际加载顺序,而非仅在“登录项”界面中拖拽排列的视觉顺序,则需深入 launchd 服务加载机制。launchd 按照依赖关系、plist 文件中的 KeepAlive/RunAtLoad 键值、以及加载时机(系统级 vs 用户级)决定真实执行序列。以下是分析该顺序的具体方法:
一、通过 launchctl list 查看已加载服务及其 PID 顺序
launchctl list 输出结果按服务加载完成时间大致排序,越靠前的条目通常越早被 launchd 初始化(注意:非严格时间序,但具备强参考性)。该命令可揭示当前会话中哪些服务已激活及加载先后线索。
1、打开终端,执行以下命令查看当前用户会话中所有已加载的代理服务:
launchctl list | grep -v "PID" | grep -v "0x"
2、观察输出列表首行与末行的服务标识符,com.apple.dock 和 com.apple.finder 通常位于前列,而用户自定义 login item 对应的服务(如 com.example.app)多出现在中后段。
3、对关键服务执行 print 命令获取其加载时间戳:
launchctl print gui/$(id -u)/com.apple.dock | grep "LastExitStatus\|StartTime"
二、检查 /Library/LaunchAgents 与 ~/Library/LaunchAgents 中 plist 的 LoadOrder 键(若存在)
部分第三方或自定义 plist 文件可能显式声明 LoadOrder 键(非标准键,需手动添加),用于提示 launchd 加载优先级。系统原生 plist 不含此键,但管理员可自行注入以影响顺序。
1、在终端中进入用户级代理目录:
cd ~/Library/LaunchAgents
2、逐个检查 plist 是否含 LoadOrder 字段:
for f in *.plist; do echo "== $f =="; plutil -p "$f" | grep -A2 -B2 LoadOrder; done
3、若发现某 plist 含 LoadOrder = 10,而另一为 LoadOrder = 30,则前者将被优先加载(需配合 launchctl bootstrap 手动重载)。
三、解析 launchd 加载阶段与目录扫描顺序
launchd 在系统启动过程中分阶段扫描特定目录,目录遍历顺序直接影响服务注册时序。该顺序由内核启动流程固化,不可更改,但可据此推断典型加载层级。
1、系统启动初期,launchd 首先加载 /System/Library/LaunchDaemons/ 下所有启用的 .plist(需 Disabled=false 或无 Disabled 键)。
2、随后加载 /Library/LaunchDaemons/ 中的 plist,此阶段常包含第三方系统守护进程。
3、用户登录后,launchd 切换至用户上下文,依次加载:
a) /System/Library/LaunchAgents/
b) /Library/LaunchAgents/
c) ~/Library/LaunchAgents/
4、最后,系统通过 SMLoginItem 机制注入“登录项”列表中的应用(即“用户与群组”中配置的 .app),此类项目实际由 loginwindow 进程调用,总晚于所有 LaunchAgents 加载完成。
四、使用 launchctl bootout 与 bootstrap 触发重载并验证顺序变化
通过手动卸载再重新注入特定服务,可强制调整其在 launchctl list 中的位置,从而实证加载顺序受加载动作时间影响。
1、选取一个测试用自定义 agent(如 ~/Library/LaunchAgents/test.plist),确保其 RunAtLoad 为 true。
2、执行卸载:
launchctl bootout gui/$(id -u)/test
3、立即重新加载:
launchctl bootstrap gui/$(id -u) ~/Library/LaunchAgents/test.plist
4、再次运行 launchctl list,该服务将出现在输出末尾,证实其为最新加载项。
五、监控 launchd 日志输出实时捕获加载事件
system.log 中记录了 launchd 每次加载服务的精确时间戳与服务名,是唯一能还原真实毫秒级顺序的日志源。
1、在终端中执行实时日志过滤:
log stream --predicate 'subsystem == "com.apple.launchd"' --info | grep -E "(loading|started) .*\.plist"
2、重启 Mac 或执行 launchctl bootout/bootstrap 操作,观察日志流中出现的首条匹配行:
"loading: path=/System/Library/LaunchDaemons/com.apple.notifyd.plist"
3、持续记录至少30秒,最早出现的 loading 行对应最先加载的服务。










