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; ?>
】
相关推荐
- Win11怎么恢复误删照片_Win11数据恢复工具
- c++怎么使用std::tuple存储多元组数据_
- Win10如何更改网络连接_Windows10以太
- VSC怎么配置PHP的Xdebug_远程调试设置步
- c++如何使用std::bitset进行位图算法_
- Win11怎么设置闹钟_Windows 11时钟应
- 如何在Golang中使用内置函数_Golangle
- Win11怎么打开注册表_Windows 11注册
- Win11怎么开启远程桌面连接_Windows11
- Mac如何创建和管理多个桌面空间_Mac高效多任务
- Win11怎么更改计算机名_Windows11系统
- PHP主流架构怎么部署到Docker_容器化流程【
- php嵌入式日志记录怎么实现_php将硬件数据写入
- php错误怎么开启_display_errors与
- Win11怎样安装搜狗输入法_Win11安装搜狗输
- c++中如何求一个数的平方根_c++ sqrt函数
- 如何在 IIS 上为 ASP.NET 6 应用排除
- Win11怎么解压RAR文件 Win11自带解压功
- 如何用正则与预处理结合精准拦截拼接式垃圾域名
- C++如何获取CPU核心数?(std::threa
- 如何在Golang中实现WebSocket广播_使
- php查询数据怎么导出csv_查询结果转csv文件
- 为什么本地php环境运行php脚本卡顿_php执行
- Win11怎么开启剪贴板历史记录_Windows1
- PHP 中如何在函数内持久修改引用变量所指向的目标
- Win11怎么关闭触摸屏_禁用Win11笔记本触摸
- SAX解析器是什么,它与DOM在处理大型XML文件
- Win10电脑怎么设置网络名称_Windows10
- Win11怎么设置默认终端应用_Windows11
- Python对象生命周期管理_创建销毁说明【指导】
- Win11怎么开启自动HDR画质_Windows1
- 如何使用Golang实现文件追加操作_向已有文件追
- Win11讲述人怎么关闭_Win11误触开启语音朗
- Linux怎么设置磁盘配额_Linux系统Quot
- Windows10系统怎么查看防火墙状态_Win1
- Win11怎么查看显卡温度 Win11任务管理器查
- Win10系统字体模糊怎么办_Windows10高
- Python对象比较与排序_魔术方法解析【教程】
- Mac的Time Machine怎么用_Mac系统
- Win11怎么禁用键盘自带键盘_Win11笔记本禁
- Win10怎样卸载DockerDesktop_Wi
- Win10如何设置双wan路由器 Win10双wa
- Avalonia如何实现跨窗口通信 Avaloni
- Win11如何设置电源计划_Win11电源计划优化
- 短链接还原php提示内存不足_调整PHP内存限制设
- Win11怎么关闭系统声音_Win11系统提示音静
- Win10如何卸载自带Edge_Win10彻底卸载
- 如何使用Golang反射创建map对象_动态生成键
- Linux怎么实现内网穿透_Linux安装Frp客
- Win11怎么关闭搜索历史_Win11清除任务栏搜


QQ客服