C++ static静态变量存在哪 C++内存分区之全局静态区详解【底层】
技术百科
穿越時空
发布时间:2026-01-24
浏览: 次 static变量存于数据段(.data)或BSS段(.bss),统称全局静态区;已初始化的进入.data,未初始化或初始化为0的进入.bss,生命周期贯穿程序运行期。
static变量到底存在内存的哪个区域
全局或命名空间作用域里的 static 变量,以及类内定义的 static 成员变量(未在类外定义的不算),都落在「数据段(Data Segment)」中已初始化的部分,也就是常说的「.data 段」;而未显式初始化的 static 变量(比如 static int x;)则进入「.bss 段」——两者合称「全局静态区」,由操作系统在进程加载时统一映射,生命周期贯穿整个程序运行期。
.data 和 .bss 的实际区别不只是“有没有初始化”
本质差异在于:.data 段内容会真实占用可执行文件体积(因为要存初始值),而 .bss 段只在 ELF 文件里记录大小,不存原始字节,加载时由内核直接清零。这意味着:
-
static int a = 42;→ 进入 .data,增加二进制文件大小 -
static int b;或static int c = 0;→ 进入 .bss,不增大文件体积 - 即使你写
static char buf[1024*1024] = {0};,编译器也可能优化进 .bss(只要全为 0) -
static const int x = 5;通常不占存储(可能被折叠进指令立即数),除非取了地址
局部 static 变量也归这里管,但行为容易误解
函数内部的 static int counter = 0; 看似“局部”,其实和全局 static 一样存在 .data/.bss,只是作用域受限。关键点:
- 初始化只发生一次(首次执行到该行时),不是每次调用都重置
- 它不放在栈上,所以不会因函数返回而销毁——这是和普通局部变量的根本区别
- 多线程环境下,C++11 起保证首次初始化是线程安全的(通过隐式锁),但后续读写仍需手动同步
- 注意:
static局部变量的初始化不是编译期常量表达式,不能用于模板非类型参数等场景
为什么调试时看不到 .bss 变量的“初始值”?
因为 .bss 在磁盘上根本没存值,GDB 加载后看到的 0 是内核清零的结果,不是从文件读出来的。如果你用 readelf -S your_program 查看节头,会发现 .bss 的 sh_type 是 SHT_NOBITS,且 sh_offset 为 0 —— 它没有对应磁盘偏移。真正容易踩的坑是:
- 误以为
static char buf[1000];会自动初始化为随机值(实际是 0) - 在嵌入式裸机环境里,启动代码必须手动清零 .bss(否则就是垃圾值)
- 链接脚本里若把 .bss 放到不可写内存区域(如 Flash),程序启动就崩溃
- 某些 LTO(Link-Time Optimization)可能把未取址的

static变量整个优化掉,导致符号找不到
全局静态区不是抽象概念,它对应 ELF 文件里真实存在的段、内存页属性(通常为 RW)、以及启动时 loader 的具体动作。理解它在哪、怎么加载、谁负责清零,比记住“存在静态区”有用得多。
# 操作系统
# 放在
# 这是
# 得多
# 加载
# 找不到
# 是从
# 首次
# 只在
# c++
# int
# 字节
# 区别
# 为什么
# 线程
# 栈
# Static
# 多线程
# 成员变量
# 作用域
# 命名空间
# char
# const
# 清零
# 局部变量
# 常量
# 落在
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- c++怎么实现大文件的分块读写_c++ 文件指针s
- Win10系统映像怎么恢复 Win10使用系统映像
- 如何理解Go指针和内存分配关系_Go Pointe
- Win11如何卸载OneDrive_Win11卸载
- Win11时间格式怎么改成12小时制 Win11时
- c++怎么使用std::tuple存储多元组数据_
- c++中的Tag Dispatching是什么_c
- Win11怎么设置鼠标宏_Win11鼠标按键自定义
- Win11怎么开启空间音效_Windows11耳机
- Win11怎么设置ipv4地址_Windows 1
- php内存溢出怎么排查_php内存限制调试与优化方
- php转mp4怎么保留字幕_php处理带字幕视频转
- Windows笔记本无法进入睡眠模式怎么办?(电源
- php485支持哪些操作系统_php485跨系统支
- 如何使用Golang实现跨域请求支持_Golang
- Win11怎么开启专注模式_Windows11时钟
- 如何解决Windows字体显示模糊的问题?(Cle
- 如何在Windows上设置闹钟和计时器_系统自带的
- php会话怎么开启_session_start函数
- Win11怎么开启剪贴板历史记录_Windows1
- Python多线程使用规范_线程安全解析【教程】
- 如何优化Golang程序CPU性能_Golang
- Python生成器表达式内存优化_惰性计算说明【指
- Python面向对象实战讲解_类与设计模式深入理解
- c# 在高并发下使用反射发射(Reflection
- C++如何使用std::async进行异步编程?(
- php怎么捕获异常_trycatch结构处理运行时
- 如何将文本文件中的竖排字符串转换为横排字符串
- Python对象比较排序规则_集合使用说明【指导】
- php文件怎么变mp4保存_php输出视频流保存为
- php嵌入式需要什么环境_搭建php+linux嵌
- Python lxml的etree和Element
- php转exe用什么工具打包快_高效打包软件推荐【
- Win11关机界面怎么改_Win11自定义关机画面
- 如何解决同一段404代码在不同主机上表现不一致的问
- Win11任务栏天气怎么关闭 Win11隐藏天气小
- C++中的constexpr和const有什么区别
- Windows如何拦截腾讯视频广告_Windows
- Win11声音太小怎么办_Windows 11开启
- Win11怎么更改默认打开方式_Win11关联文件
- 如何在Golang中实现WebSocket广播_使
- Windows怎样拦截QQ浏览器广告_Window
- VSC怎么创建PHP项目_从零开始搭建项目的步骤【
- Win11开机自检怎么关闭_跳过Win11开机磁盘
- php485返回数据不完整怎么办_php485数据
- 如何在JavaScript中动态拼接PHP的bas
- windows如何禁用驱动程序强制签名_windo
- Python字符串处理进阶_切片方法解析【指导】
- c# 在ASP.NET Core中管理和取消后台任
- Python如何创建带属性的XML节点


QQ客服