0

0

C++程序的启动和终止过程是怎样的?C++ main函数前后详解【运行时】

尼克

尼克

发布时间:2025-12-20 18:25:02

|

232人浏览过

|

来源于php中文网

原创

c++kquote>C++程序执行包含main前初始化、main中逻辑、main后清理三阶段:main前完成加载、运行时准备、全局/静态对象构造;main中执行用户代码并隐式返回0;main后逆序析构静态对象、调用atexit函数、关闭流并由OS回收资源。

c++程序的启动和终止过程是怎样的?c++ main函数前后详解【运行时】

一个C++程序从双击图标(或命令行输入)开始,到进程彻底退出,并不只是执行 main() 那几行代码那么简单。背后有编译器、链接器和运行时库(如 libc++ 或 MSVCRT)协同完成的一整套初始化与清理流程。

main 之前:全局对象构造与运行时准备

main() 函数第一行代码执行前,系统已完成以下关键步骤:

  • 操作系统加载可执行文件:将代码段、数据段映射进内存,设置和堆的初始状态,跳转到入口点(通常是 _start,不是 main
  • C 运行时初始化:设置标准输入/输出/错误流(stdin/stdout/stderr),初始化环境变量 environ,处理命令行参数 argc/argv
  • 全局/静态对象构造:按定义顺序(同一编译单元内)调用所有命名空间作用域static 局部变量的构造函数;跨编译单元顺序未定义,但可通过 init_priority(GCC)或 __attribute__((constructor)) 控制
  • atexit 注册函数暂存:虽尚未执行,但已准备好后续注册的终止处理函数的调用链表

main 执行中:用户逻辑与资源管理

main() 是用户代码的起点,但它本身是被调用的函数——返回值会作为进程退出码传给操作系统。注意几点:

  • 可以写成 int main()int main(int argc, char* argv[]),其他签名(如 void main())非标准,不可移植
  • 未显式 return 时,C++11 起隐式等价于 return 0;(成功退出)
  • 局部静态对象首次进入作用域时构造,生命周期持续到程序结束;其析构时机在 main 返回后、全局析构前
  • 不建议在 main 中用 std::exit()std::abort() 提前退出——它们会跳过局部对象析构(但会执行 atexit 函数)

main 之后:析构、清理与终止

main() 返回(或调用 std::exit())后,运行时依次执行:

TalkMe
TalkMe

与AI语伴聊天,练习外语口语

下载

立即学习C++免费学习笔记(深入)”;

  • 局部静态对象析构:按构造的逆序销毁所有局部静态变量(含函数内 static 对象)
  • 全局/静态对象析构:按构造的逆序调用所有命名空间作用域对象的析构函数
  • atexit 注册函数调用:按注册的逆序执行所有通过 std::atexit()std::at_quick_exit() 注册的函数
  • C 运行时收尾:刷新并关闭 stdout/stderrstdin 通常只读,不关闭),释放内部缓冲区
  • 操作系统回收资源:内核释放进程占用的内存页、文件描述符、信号处理状态等,返回退出码给父进程

特殊情形与注意事项

有些行为容易被忽略,却影响程序健壮性:

  • 静态对象析构依赖问题:若 A 的析构函数访问了 B 的静态成员,而 B 构造晚于 A,则 B 可能已被析构——这是“静态初始化顺序惨案”的后半场
  • 线程安全的 main 前后:C++11 起,静态局部变量的首次初始化是线程安全的;但全局对象构造本身不保证多线程安全
  • std::quick_exit() 的区别:它跳过全局析构和大多数清理,只执行 at_quick_exit 注册函数,适合紧急终止(如崩溃恢复)
  • 嵌入式或 freestanding 环境:无 main、无运行时初始化,入口是自定义符号(如 _start),需手动管理一切

基本上就这些。理解 main 前后的完整链条,不是为了炫技,而是写出更可控的初始化逻辑、避免析构时访问悬空资源、以及在服务程序或长期运行系统中做好资源兜底。不复杂,但容易忽略。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

318

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

538

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

52

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

175

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

97

2025.11.27

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

392

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

572

2023.08.10

PHP WebSocket 实时通信开发
PHP WebSocket 实时通信开发

本专题系统讲解 PHP 在实时通信与长连接场景中的应用实践,涵盖 WebSocket 协议原理、服务端连接管理、消息推送机制、心跳检测、断线重连以及与前端的实时交互实现。通过聊天系统、实时通知等案例,帮助开发者掌握 使用 PHP 构建实时通信与推送服务的完整开发流程,适用于即时消息与高互动性应用场景。

3

2026.01.19

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Rust 教程
Rust 教程

共28课时 | 4.6万人学习

PostgreSQL 教程
PostgreSQL 教程

共48课时 | 7.4万人学习

Git 教程
Git 教程

共21课时 | 2.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号