4. C++ static关键字


一、static 到底是个什么东西?(一句话先立住)

static 的核心作用只有两个:

1️⃣ 改变"生命周期" → 活得更久

2️⃣ 改变"作用域 / 可见性" → 看得更少

⚠️ 注意:
static ≠ 常量static ≠ 线程安全static ≠ 全局变量


二、最基础的 static(函数里的 static 变量)

2.1 先看代码(你一定写过但没想明白)

cpp 复制代码
#include <iostream>
using namespace std;

void foo() {
    static int count = 0;
    count++;
    cout << count << endl;
}

int main() {
    foo();
    foo();
    foo();
}

输出:

复制代码
1
2
3

2.2 这段代码 到底发生了什么?(非常重要)

❌ 错误理解(新手 90% 会这么想)

每次进入 foo()count 都重新创建

✅ 真相(面试 + 实战要这么说)

  • count 只在程序启动时初始化一次
  • 变量存放在 静态存储区(不是栈!)
  • foo() 每次调用只是 重复使用同一个变量

📌 内存层面:

变量 在哪
普通局部变量
static 局部变量 静态区
全局变量 静态区

2.3 static 局部变量 = "函数私有的全局变量"

✔ 生命周期:整个程序

✔ 作用域:只在函数内可见

这点非常常用,比如:

  • 计数器
  • 单例
  • 延迟初始化
  • 状态缓存

三、文件作用域的 static(C/C++ 老兵最爱)

3.1 不加 static 的全局变量(危险)

cpp 复制代码
// a.cpp
int g_value = 10;
cpp 复制代码
// b.cpp
int g_value = 20;   // ❌ 链接冲突

结果:

👉 multiple definition of g_value


3.2 加上 static(内部链接)

cpp 复制代码
// a.cpp
static int g_value = 10;
cpp 复制代码
// b.cpp
static int g_value = 20;

✔ 编译通过

✔ 互不影响


3.3 static 在文件级别的真正含义

static = 只在当前编译单元可见

📌 编译单元 = 一个 .cpp + 它包含的 .h

所以:

cpp 复制代码
static void helper() {
    ...
}

✔ 只能被当前 .cpp 使用

不会污染全局符号表

📌 在大型项目里,这是必须的习惯


四、类中的 static(C++ 的核心重头戏)

这部分 极其重要,Qt / 面试 / 工程天天用。


4.1 static 成员变量(所有对象共享)

示例

cpp 复制代码
class Person {
public:
    static int count;
    Person() {
        count++;
    }
};

int Person::count = 0;   // ❗必须类外定义
cpp 复制代码
int main() {
    Person p1;
    Person p2;
    cout << Person::count << endl; // 2
}

4.2 关键认知(一定记牢)

❌ 每个对象一份

整个类只有一份

📌 内存模型:

复制代码
Person::count  ──► 静态区(1 份)
p1
p2
p3

4.3 为什么 static 成员变量要类外定义?

因为:

static 成员变量不属于任何对象

所以:

  • 类里只是"声明"
  • 真正的存储空间要单独分配

📌 C++17 以后可以这样写(面试加分):

cpp 复制代码
class Person {
public:
    inline static int count = 0;
};

五、static 成员函数(非常容易踩坑)

5.1 代码示例

cpp 复制代码
class Math {
public:
    static int add(int a, int b) {
        return a + b;
    }
};

调用方式:

cpp 复制代码
int x = Math::add(1, 2);

5.2 static 成员函数的本质

能做 不能做
访问 static 成员 ❌ 访问非 static 成员
没有 this 指针 ❌ 使用 this

为什么?

因为:

static 成员函数不属于任何对象


5.3 常见错误(面试必考)

cpp 复制代码
class A {
    int x;
public:
    static void foo() {
        x = 10; // ❌ 错误
    }
};

✔ 正确做法:

cpp 复制代码
static void foo(A& a) {
    a.x = 10;
}

六、static 在构造 / 析构 / 单例中的使用(实战)

6.1 经典单例(C++11 以后最推荐)

cpp 复制代码
class Singleton {
public:
    static Singleton& instance() {
        static Singleton s; // 线程安全(C++11)
        return s;
    }
private:
    Singleton() {}
};

为什么靠谱?

  • 懒加载
  • 线程安全
  • 自动析构
  • 无内存泄漏

📌 Qt / STL 内部大量用这种模式


七、static 和 const 的区别(非常容易混)

对比 static const
生命周期 程序全程 取决于定义位置
本质 存储期 / 可见性 只读语义
是否共享 不一定

👉 可以同时使用

cpp 复制代码
static const int MAX = 100;

八、static 常见误区(你以后一定会踩)

❌ 以为 static 是线程安全

❌ 滥用 static 造成"隐式全局变量"

❌ 在头文件里定义 static 对象

❌ 用 static 替代设计(耦合爆炸)


九、什么时候该用 / 不该用 static(工程经验)

✅ 该用

  • 工具函数(工具类)
  • 类级别计数
  • 单例
  • 模块内私有变量

❌ 不该用

  • 需要多实例状态
  • 业务对象
  • 需要解耦 / 测试的模块

十、面试一句话总结(直接背)

static 改变的是变量或函数的生命周期和可见性,在 C++ 中既用于控制链接属性,也用于类级别共享数据和行为。


相关推荐
炘爚17 小时前
C语言(文件操作)
c语言·开发语言
阿蒙Amon17 小时前
C#常用类库-详解SerialPort
开发语言·c#
凸头18 小时前
CompletableFuture 与 Future 对比与实战示例
java·开发语言
wuqingshun31415918 小时前
线程安全需要保证几个基本特征
java·开发语言·jvm
君义_noip18 小时前
信息学奥赛一本通 1952:【10NOIP普及组】三国游戏 | 洛谷 P1199 [NOIP 2010 普及组] 三国游戏
c++·信息学奥赛·csp-s
Moksha26218 小时前
5G、VoNR基本概念
开发语言·5g·php
jzlhll12318 小时前
kotlin Flow first() last()总结
开发语言·前端·kotlin
W.D.小糊涂18 小时前
gpu服务器安装windows+ubuntu24.04双系统
c语言·开发语言·数据库
旖-旎18 小时前
二分查找(x的平方根)(4)
c++·算法·二分查找·力扣·双指针
用头发抵命18 小时前
Vue 3 中优雅地集成 Video.js 播放器:从组件封装到功能定制
开发语言·javascript·ecmascript