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++ 中既用于控制链接属性,也用于类级别共享数据和行为。


相关推荐
橘颂TA8 小时前
【Linux】死锁四条件的底层逻辑:从锁冲突到 STL 组件的线程安全实践(Ⅵ)
linux·运维·服务器·c++·死锁
C++ 老炮儿的技术栈8 小时前
什么是通信规约
开发语言·数据结构·c++·windows·算法·安全·链表
@大迁世界8 小时前
TypeScript 的本质并非类型,而是信任
开发语言·前端·javascript·typescript·ecmascript
栗子叶8 小时前
Java对象创建的过程
java·开发语言·jvm
Amumu121388 小时前
React面向组件编程
开发语言·前端·javascript
IT=>小脑虎9 小时前
Python零基础衔接进阶知识点【详解版】
开发语言·人工智能·python
wjs20249 小时前
C 标准库 - `<float.h>》详解
开发语言
zfj3219 小时前
CyclicBarrier、CountDownLatch、Semaphore 各自的作用和用法区别
java·开发语言·countdownlatch·semaphore·cyclicbarrier
张np9 小时前
java基础-ConcurrentHashMap
java·开发语言