static和const的作用

本文详细解析了C/C++编程语言中的static与const关键字的作用。static关键字涉及变量和函数的作用域及生命周期,而const关键字则用于限定变量或参数的可变性。文中列举了两种关键字在不同场景下的具体应用。

static关键字至少有下列n个作用:

  (1)函数体内static变量的作用范围为该函数体,不同于auto变量,该变量的内存只被分配一次,因此其值在下次调用时仍维持上次的值;

  (2)在模块内的static全局变量可以被模块内所用函数访问,但不能被模块外其它函数访问;

  (3)在模块内的static函数只可被这一模块内的其它函数调用,这个函数的使用范围被限制在声明它的模块内;

  (4)在类中的static成员变量属于整个类所拥有,对类的所有对象只有一份拷贝;

  (5)在类中的static成员函数属于整个类所拥有,这个函数不接收this指针,因而只能访问类的static成员变量。


const关键字至少有下列n个作用:

  (1)欲阻止一个变量被改变,可以使用const关键字。在定义该const变量时,通常需要对它进行初始化,因为以后就没有机会再去改变它了;

  (2)对指针来说,可以指定指针本身为const,也可以指定指针所指的数据为const,或二者同时指定为const;

  (3)在一个函数声明中,const可以修饰形参,表明它是一个输入参数,在函数内部不能改变其值;

  (4)对于类的成员函数,若指定其为const类型,则表明其是一个常函数,不能修改类的成员变量;

  (5)对于类的成员函数,有时候必须指定其返回值为const类型,以使得其返回值不为“左值”。例如:

  operator*的返回结果必须是一个const对象。如果不是,这样的变态代码也不会编译出错:

  操作(a * b) = c显然不符合编程者的初衷,也没有任何意义。 

<think>我们讨论的是C/C++中的`const static``static const`。首先,需要明确的是,在C/C++中,`const static``static const`是等价的,没有区别。它们都表示一个静态的常量(即只读的静态变量)。不过,在书写顺序上,有些人习惯写`static const`,而有些人习惯写`const static`,但编译器将其视为相同。 ### 1. 含义 - **`static`**:表示静态存储期内部链接(对于全局变量)或静态存储期(对于局部变量)。 - 在函数内部(局部变量):变量在程序运行期间只初始化一次,且生命周期持续到程序结束。每次调用函数时,该变量保持上一次的值。 - 在全局作用域:变量或函数的作用域被限制在当前文件(即内部链接),其他文件无法访问。 - **`const`**:表示变量是只读的,即常量。一旦初始化后,其值不能被修改。 当两者结合时,即`static const`或`const static`,表示一个静态的常量。 ### 2. 使用场景 #### (1) 全局作用域 在全局作用域中,`static const`(或`const static`)用于定义文件作用域内的常量。由于`static`的修饰,该常量只在当前文件可见,避免了与其他文件的命名冲突。同时,`const`确保其值不可修改。 示例: ```c // 在file1.cpp中 static const int MAX_SIZE = 100; // 或 const static int MAX_SIZE = 100; // 在file2.cpp中 extern const int MAX_SIZE; // 错误!无法访问file1.cpp中的MAX_SIZE,因为static使其具有内部链接 ``` #### (2) 局部作用域(函数内部) 在函数内部,`static const`用于定义只初始化一次且不可修改的局部变量。这种变量在程序运行期间一直存在,但只能在函数内部访问。 示例: ```c void func() { static const int local_const = 42; // 初始化只发生一次 // local_const = 43; // 错误,因为const修饰,不可修改 // 每次调用func,local_const都是42,且不会重新初始化 } ``` #### (3) 类成员(C++) 在C++中,`static const`可以用于类的静态常量成员。此时,该成员属于整个类,而不是某个对象。它必须在类外定义(除非是整型或枚举类型,且C++11后允许在类内初始化)。 示例: ```c++ class MyClass { public: static const int MAX = 100; // 声明,如果是整型常量,可以在类内初始化(C++11) }; // 在类外定义(如果需要在类外分配存储空间,则必须定义) const int MyClass::MAX; // 如果类内已初始化,这里可以不带初始化值,但通常还是需要定义(有些编译器要求) ``` 注意:在C++中,`const static`的写法也是可以的,但通常使用`static const`。 ### 3. 与单独使用`const`或`static`的区别 - **单独使用`static`(非`const`)**:变量是静态的,但值可以修改(非常量)。 - **单独使用`const`(非`static`)**: - 在全局作用域:默认具有外部链接(除非在匿名命名空间),即其他文件可以通过`extern`声明来访问。但在C++中,全局`const`变量默认具有内部链接(除非用`extern`显式指定外部链接)。 - 在局部作用域:每次函数调用都会重新初始化,且值不可修改,但生命周期仅限于函数调用期间(实际上,由于是常量,编译器可能将其优化为立即数,不占用栈空间)。 ### 4. 内存存储 - 对于全局的`static const`:通常存储在只读数据段(`.rodata`)或直接嵌入代码中(作为立即数)。 - 对于局部的`static const`:存储在静态存储区,但只初始化一次,且不可修改。 ### 5. 注意事项 - **初始化**:`static const`必须在声明时初始化(因为`const`要求初始化,且`static`要求只初始化一次)。 - **C与C++的差异**:在C中,全局`const`变量默认具有外部链接,而C++中默认具有内部链接(因此C++中全局`const`变量相当于隐式加了`static`)。但为了代码清晰,建议显式使用`static`。 ### 示例代码 ```c // C++示例:文件作用static const int g_value = 10; // 文件内可见,不可修改 void foo() { static const int local_value = 20; // 静态常量,只初始化一次 // ... } // C++类成员 class Test { public: static const int MAX = 100; // 类内声明(整型常量) }; const int Test::MAX; // 类外定义(有些编译器要求,即使类内已初始化) ``` ### 相关问题 1. 在C++中,为什么有时候需要在类外定义静态常量成员? 2. `static const`成员变量`constexpr`成员变量有什么区别? 3. 如何在不同文件之间共享常量定义(即具有外部链接的常量)? 4. 在C++中,全局`const`变量默认的内部链接是如何影响程序的?</think>### `const static` 与 `static const` 详解 在 C/C++ 中,`const static` `static const` **本质相同**,两者无功能差异。编译器会将其视为同一种声明,区别仅在于关键字顺序(书写习惯)。以下是核心知识点: --- ### 一、核心含义 #### 1. **`static` 的作用** - **存储期控制**: 变量在程序整个生命周期存在(存储在静态存储区),而非栈或堆[^1]。 - **作用域控制**: - **全局变量/函数**:限制作用域为当前文件(内部链接) - **局部变量**:保持值在函数调用间持久化 - **类成员**:属于类而非实例(所有对象共享) #### 2. **`const` 的作用** - **只读性**:变量初始化后不可修改 - **编译期检查**:防止意外写入(如 `str[4]='x'` 会触发编译错误)[^4] #### 3. **组合效果** `static const` 或 `const static` 表示: - **静态存储**(程序生命周期) - **只读属性**(不可修改) - 作用域由声明位置决定 --- ### 二、使用场景与示例 #### 1. **全局常量(文件作用域)** ```c++ // File: config.cpp static const int MAX_TEMP = 150; // 文件内可见,外部无法链接 // 等效写法:const static int MAX_TEMP = 150; ``` - **作用**:避免全局命名冲突,作为文件私有常量 - **内存**:存储在只读数据段(`.rodata`) #### 2. **函数内持久化常量** ```c++ void read_sensor() { static const float CALIBRATION_FACTOR = 1.023f; // 只初始化一次 // CALIBRATION_FACTOR = 1.1f; // 错误:const禁止修改 } ``` - **特点**: - 首次调用时初始化,后续调用保留值 - 线程安全(C++11 起保证初始化线程安全) #### 3. **类静态常量(C++)** ```c++ class TemperatureMonitor { public: static const int SAFE_LIMIT = 85; // 声明(整型可直接初始化) }; // 类外定义(非整型或需取地址时必需) const int TemperatureMonitor::SAFE_LIMIT; ``` - **用途**:所有对象共享的类级别常量 - **限制**: - C++ 中仅整型(`int`, `char` 等)可在类内直接初始化 - 非整型需在类外定义(如 `static const double PI;`) --- ### 三、关键区别对比 | 特性 | 单独 `static` | 单独 `const` | `static const`/`const static` | |---------------------|-----------------------|-----------------------|-------------------------------------| | **存储位置** | 静态存储区 | 栈或常量区 | 静态存储区(只读段) | | **生命周期** | 程序整个生命周期 | 作用域结束时销毁 | 程序整个生命周期 | | **修改权限** | 可修改 | **不可修改** | **不可修改** | | **多文件共享** | 需 `extern` 声明 | 默认内部链接(C++) | 文件内私有(`static` 作用) | | 典型用例 | 计数器、缓存 | 函数参数保护 | 配置常量、数学常数 | --- ### 四、实际应用建议 1. **优先使用 `static const`** - 统一书写顺序(业界惯例:`static const`) - 例:`static constexpr auto PI = 3.14159;`(C++11 起) 2. **避免全局非 `static` 常量** ```c++ // 不推荐(C++中默认内部链接,但C中为外部链接) const int GLOBAL_CONST = 100; // 推荐:显式控制作用static const int FILE_SCOPED_CONST = 100; ``` 3. **C 与 C++ 差异处理** - **C 语言**:全局 `const` 默认外部链接,需加 `static` 限制作用域 - **C++**:全局 `const` 默认内部链接,`static` 可省略(但显式添加更清晰) --- ### 五、相关问题 1. `static const` 成员变量在 C++ 中为什么需要类外定义? 2. 如何在不同 C++ 文件间共享常量定义? 3. `constexpr` `static const` 在性能上有何差异? 4. 多线程环境下使用静态局部常量是否安全? > 通过合理使用 `static const`,可提升代码安全性(`const`)并优化资源管理(`static`),是嵌入式系统性能敏感场景的常用技术[^1][^2][^4]。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值