嵌入式的C/C++:深入理解 static、const 与 volatile 的用法与特点

目录

一、static

1、static 修饰局部变量

 2、 static 修饰全局变量

3、static 修饰函数

4、static 修饰类成员

5、小结

二、const

1、const 修饰普通变量

2、const 修饰指针

3、const 修饰函数参数

4. const 修饰函数返回值

5. const 修饰类成员

6. const 与 #define 的比较

7. 小结

三、 volatile

1、volatile 的作用

2、volatile 的典型应用场景

3、volatile 的特性与限制

4、volatile 的用法

5、小结


在嵌入式的C/C++ 编程中,关键字不仅仅是语法结构的一部分,更是语言核心特性的体现。staticconstvolatile 是三个常见且重要的关键字,广泛应用于变量管理、优化控制、代码安全性和硬件编程等领域。然而,很多开发者在使用它们时,往往只了解表面作用,而忽视了深入理解可能带来的性能优化和代码维护收益。本篇博客将通过细致的分类讲解和实用的示例,带你全面掌握这三个关键字的用法、特性和应用场景。

一、static

static 关键字有多种用途。在函数内部声明的变量前使用 static 关键字,可以让该变量在整个程序运行期间都保持其值,而不是在每次调用函数时重新初始化。对于全局变量或函数,static 可以限制它们的作用域到声明它们的文件内,即其他文件无法访问这些变量或函数。

1、static 修饰局部变量

作用:将局部变量的 生命周期 扩展为整个程序的运行期间,但 作用域 仍局限于函数内部

特点:初始化只会发生一次。再次调用函数时,保留变量上一次的值。

示例:

#include <stdio.h>
void counter() {static int count = 0;  // 静态局部变量,初始化只执行一次count++;printf("Count: %d\n", count);
}int main() {counter();  // 输出:Count: 1counter();  // 输出:Count: 2counter();  // 输出:Count: 3return 0;
}

分析

  1. count 是静态局部变量,第一次调用时初始化为 0。
  2. 每次调用 counter 函数后,count 的值都会被保留,而不是销毁。
  3. 如果没有 staticcount 每次调用都会重新初始化为 0。

 2、 static 修饰全局变量

作用:将全局变量的 作用域 限制在当前文件中,使得其他文件无法直接访问该变量。

特点:全局变量默认具有整个程序可见性,但加上 static 后,仅对声明它的文件可见。有助于模块化和避免命名冲突。

示例:

// file1.c
#include <stdio.h>
static int global_var = 100; // 静态全局变量,仅限于本文件void display() {printf("global_var: %d\n", global_var);
}// file2.c
#include <stdio.h>
extern int global_var; // 错误!无法访问 file1.c 中的静态全局变量int main() {display(); // 只能通过函数间接访问return 0;
}

分析

  1. global_var 是静态全局变量,仅 file1.c 可访问,file2.c 无法通过 extern 引用。
  2. 模块化设计中,通过隐藏不必要的全局变量,减少模块间的耦合。

3、static 修饰函数

作用:将函数的作用域限制在当前文件中,避免外部文件调用该函数,起到“私有化”的作用。

特点:函数默认具有外部可见性,加上 static 后仅对当前文件可见。有助于避免命名冲突。

示例:

// file1.c
#include <stdio.h>
static void privateFunction() {printf("This is a static function.\n");
}void publicFunction() {privateFunction(); // 内部可以正常调用
}// file2.c
extern void privateFunction(); // 错误!无法访问 file1.c 中的静态函数int main() {publicFunction(); // 通过非静态函数间接调用return 0;
}

分析

  1. privateFunction 是静态函数,仅 file1.c 内部可调用,file2.c 无法通过 extern 声明使用。
  2. 这种机制可以防止外部文件误调用函数,起到保护作用。

4、static 修饰类成员

4.1 静态成员变量

特点:属于整个类而非某个对象。在所有对象中共享,仅在程序中初始化一次。

示例:

#include <iostream>
class MyClass {
public:static int count; // 静态成员变量MyClass() { count++; }
};int MyClass::count = 0; // 静态成员变量初始化int main() {MyClass obj1, obj2, obj3;std::cout << "Object count: " << MyClass::count << std::endl; // 输出:3return 0;
}
4.2 静态成员函数

特点:不能访问非静态成员(因为没有对象实例)。可通过类名直接调用,而无需实例化对象。

示例:

#include <iostream>
class MyClass {
public:static void display() {std::cout << "This is a static member function." << std::endl;}
};int main() {MyClass::display(); // 静态成员函数直接通过类名调用return 0;
}

5、小结

static 关键字在模块化、性能优化和作用域管理中起到非常重要的作用。

二、const

const 关键字用于指定一个对象是常量,即它的值不能通过普通的赋值操作来改变。它可以应用于变量、指针、函数参数等。用于定义不可修改的值具有只读属性的变量。它在程序设计中用于增强代码的安全性和可维护性。

1、const 修饰普通变量

作用:定义一个只读的变量,不能对其赋新值。

特点:变量的值在程序中固定。编译器会在尝试修改 const 变量时报错。

示例:

#include <stdio.h>
int main() {const int x = 10;  // 定义只读变量printf("x = %d\n", x);// x = 20; // 错误:不能修改 const 变量return 0;
}

注意const 修饰的变量必须初始化,否则会报错

2、const 修饰指针

在指针的定义中,const 的位置决定了指针的只读属性是指针本身还是指针指向的值

2.1 指向的值是只读的

const int *p = &x; // 或 int const *p = &x;

含义:指针指向的值不能修改,但指针本身可以改变指向的地址。

示例

#include <stdio.h>
int main() {int x = 10, y = 20;const int *p = &x;  // 指针指向的值不可修改// *p = 15; // 错误:不能修改 p 指向的值p = &y;   // 合法:可以修改 p 的指向printf("p points to: %d\n", *p);return 0;
}

2.2 指针本身是只读的

int * const p = &x;

含义:指针本身不能修改指向的地址,但指针指向的值可以改变。

示例

#include <stdio.h>
int main() {int x = 10;int *const p = &x;  // 指针本身不可修改*p = 20;            // 合法:可以修改 p 指向的值// p = &y;          // 错误:不能改变指针的指向printf("x = %d\n", *p);return 0;
}

2.3 指针本身和指向的值都是只读的

const int * const p = &x;

含义:指针本身和指向的值都不可修改。

3、const 修饰函数参数

作用:保护函数的输入参数,防止在函数内部被修改。

应用场景:适用于传入参数的值不应被修改的情况(例如输入只读配置、避免意外修改等)。

3.1 修饰值传递参数

void func(const int x) {// x = 20; // 错误:x 是只读的printf("x = %d\n", x);
}
int main() {func(10);return 0;
}

3.2 修饰指针参数

如果函数需要读取指针指向的值,但不能修改该值:

void display(const int *p) {// *p = 20; // 错误:不能修改 p 指向的值printf("Value: %d\n", *p);
}

如果函数不能修改指针本身的地址:

void display(int * const p) {// p = &y; // 错误:不能修改指针 p 本身的地址*p = 20; // 合法:可以修改 p 指向的值
}

4. const 修饰函数返回值

作用:防止函数返回值被修改。

4.1 修饰返回普通值

const int getValue() {return 10;
}
int main() {const int x = getValue();// x = 20; // 错误:不能修改 x 的值return 0;
}

4.2 修饰返回指针

如果函数返回一个指针,但指针指向的值不可修改:

const int* getPointer() {static int x = 10;return &x;
}
int main() {const int *p = getPointer();// *p = 20; // 错误:不能修改 p 指向的值return 0;
}

5. const 修饰类成员

5.1 修饰类成员变量

作用:类的成员变量定义为只读,必须在构造函数初始化列表中初始化。

#include <iostream>
class MyClass {
public:const int value; // const 成员变量MyClass(int v) : value(v) {} // 必须在构造函数初始化列表中初始化
};int main() {MyClass obj(10);// obj.value = 20; // 错误:不能修改 const 成员变量std::cout << "Value: " << obj.value << std::endl;return 0;
}

5.2 修饰类成员函数

作用:表示该函数不会修改类的成员变量。

语法:在函数定义后加 const

#include <iostream>
class MyClass {
private:int data;
public:MyClass(int d) : data(d) {}int getData() const {  // const 成员函数return data;}// void setData(int d) const { data = d; } // 错误:const 函数不能修改成员变量
};int main() {MyClass obj(10);std::cout << "Data: " << obj.getData() << std::endl;return 0;
}

6. const#define 的比较

示例对比:

#define PI 3.14
const double pi = 3.14;int main() {// PI = 3.15; // 错误:#define 定义的值是文本替换,不是变量// pi = 3.15; // 错误:const 定义的值不能修改return 0;
}

7. 小结

const 的作用:

  1. 保护变量:限制变量或指针的可修改性,增强安全性。

  2. 优化函数:保护函数参数,避免意外修改。

  3. 增强表达力:通过 const 声明,提高代码的可读性和维护性。

  4. C++ 专用特性:在类中可修饰成员变量和成员函数,支持更复杂的只读逻辑。

const 是代码中保证不变性的强有力工具,在安全性、优化和可维护性方面至关重要。

三、 volatile

volatile 关键字用来修饰那些可能被意想不到地改变的变量,例如硬件寄存器中的值或并发线程中共享的变量。它告诉编译器不要对涉及这些变量的操作进行优化,确保每次读取都是从内存中读取最新的值。

1、volatile 的作用

防止编译器优化
编译器在优化代码时,可能会将变量的值缓存在寄存器中,导致程序无法感知变量的实时变化。volatile 保证变量的值总是从内存中读取,而不是从寄存器缓存读取。

编译器在优化代码时,可能会做以下处理:

将变量的值缓存到寄存器中,避免重复访问内存。

在循环中,认为变量值不变,将其优化为常量。

volatile 禁止编译器对变量进行这些优化。

确保正确性
对于可能被其他线程、中断服务程序(ISR)、硬件设备等修改的变量,volatile 确保程序访问的是最新值。

2、volatile 的典型应用场景

2.1 多线程编程

当一个变量可能被多个线程修改时,需要用 volatile 声明,以防止编译器优化。

volatile int flag = 0;void thread1() {while (flag == 0) {// 等待其他线程修改 flag}// 继续执行
}
void thread2() {flag = 1;  // 修改 flag
}

如果没有 volatile,编译器可能会将 flag == 0 优化为一个死循环,因为它认为 flag 的值不会改变。

2.2 硬件寄存器访问

与硬件交互时,寄存器的值可能在程序之外发生变化,必须使用 volatile 确保程序访问到最新值。

#define STATUS_REGISTER *((volatile int *)0x40000000)void checkStatus() {while ((STATUS_REGISTER & 0x01) == 0) {// 等待硬件设置状态寄存器的第 0 位}// 状态已改变
}
2.3 中断服务程序 (ISR)

中断服务程序可能会修改主程序中的变量,因此这些变量需要声明为 volatile

volatile int timer_flag = 0;void ISR() {timer_flag = 1;  // 中断发生时修改变量
}int main() {while (timer_flag == 0) {// 等待中断}// 中断已发生return 0;
}

3、volatile 的特性与限制

作用:告诉编译器变量可能会被外部事件(硬件或线程)修改,因此每次访问变量时必须从内存中读取。

内存可见性:volatile 保证了 单线程环境 中的变量内存可见性,即从内存中读取最新值,但 不提供线程安全性(需要配合锁或原子操作)。

无法保证线程安全
volatile 仅保证变量值从内存中读取,但无法保证多个线程对同一变量的原子操作。例如:

volatile int counter = 0;void increment() {counter++;  // 非线程安全,可能发生数据竞争
}

 此时需要使用 互斥锁

无法控制操作顺序
例如在多核处理器中,内存屏障(memory barrier)需要单独使用,volatile 无法确保操作的顺序。

4、volatile 的用法

4.1修饰普通变量

volatile int counter = 0;void modifyCounter() {counter++;  // 确保每次操作都从内存读取
}

4.2 修饰指针

指针本身是 volatile

int * volatile p;

含义:指针地址可能发生变化,但指针指向的值可以改变。

指针指向的值是 volatile

volatile int *p;

含义:指针指向的值是可变的,必须从内存读取。

指针本身和指向的值都是 volatile

volatile int * volatile p;

含义:指针本身和指针指向的值都可能被外部修改。

4.3缓存优化问题

假设没有使用 volatile

int flag = 0;void waitForFlag() {while (flag == 0) {// 循环等待}
}

在某些编译器中,while (flag == 0) 可能被优化为:

if (flag == 0) {while (true) {} // 死循环
}

因为编译器假定 flag 不会被外部修改。

添加 volatile 后:

volatile int flag = 0;void waitForFlag() {while (flag == 0) {// 每次读取最新的 flag 值}
}

4.4硬件寄存器问题

#define STATUS_REG *((volatile int *)0x40000000)void pollStatus() {while ((STATUS_REG & 0x01) == 0) {// 等待硬件设置状态寄存器的第 0 位}
}

如果没有 volatile,编译器可能优化为:

int reg = STATUS_REG;
while ((reg & 0x01) == 0) {// 死循环,无法感知硬件变化
}

5、小结

注意事项:

  1. 对于需要确保线程安全的操作,需要配合 互斥锁原子操作

  2. 硬件编程中,寄存器值必须声明为 volatile,否则可能导致严重的逻辑错误。

volatile 是嵌入式系统和并发编程中不可或缺的关键字,在适当的场景下使用它能够显著提高代码的正确性和健壮性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/60776.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

《Python基础》之列表推导式(列表生成式)

目录 简介 用法 1、基本列表推导式 结果如下 2、待条件的列表推导式 结果如下 3、嵌套列表推导式 结果如下 4、使用函数 结果如下 5、 处理字符串 结果如下 总结 优点 注意事项 简介 列表推导式&#xff08;List Comprehension&#xff09;是Python中一种简洁且…

qt QDateTime详解

1. 概述 QDateTime 是 Qt 框架中用于处理日期和时间的类。它将 QDate 和 QTime 组合在一起&#xff0c;提供了日期时间的统一处理方案。QDateTime 可以精确到毫秒&#xff0c;并支持时区处理。 2. 重要方法 构造函数: QDateTime() 构造无效的日期时间 QDateTime(const QDa…

Neural Magic 发布 LLM Compressor:提升大模型推理效率的新工具

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

jQuery-Word-Export 使用记录及完整修正文件下载 jquery.wordexport.js

参考资料&#xff1a; jQuery-Word-Export导出word_jquery.wordexport.js下载-CSDN博客 近期又需要自己做个 Html2Doc 的解决方案&#xff0c;因为客户又不想要 Html2pdf 的下载了&#xff0c;当初还给我费尽心思解决Html转pdf时中文输出的问题&#xff08;html转pdf文件下载之…

第8章 文件上传与下载

第八章 文件上传与下载 8.1 文件上传 使用SpringMVC6版本&#xff0c;不需要添加以下依赖&#xff1a; <dependency><groupId>commons-fileupload</groupId><artifactId>commons-fileupload</artifactId><version>1.5</version> …

sql工具!好用!爱用!

SQLynx的界面设计简洁明了&#xff0c;操作逻辑清晰易懂&#xff0c;没有复杂的图标和按钮&#xff0c;想对哪部分操作就在哪里点击右键&#xff0c;即使你是数据库小白也能轻松上手。 尽管SQLynx是一款免费的工具&#xff0c;但是它的功能却丝毫不逊色于其他付费产品&#xff…

Pytest-Bdd-Playwright 系列教程(13):钩子(hooks)

Pytest-Bdd-Playwright 系列教程&#xff08;13&#xff09;&#xff1a;钩子&#xff08;hooks&#xff09; 前言一、什么是钩子&#xff1f;二、Pytest-Bdd 提供的钩子一览三、钩子用法详解1. pytest_bdd_before_scenario2. pytest_bdd_after_scenario3. pytest_bdd_before_s…

竞赛经验:关于不记得字母表,如何知道字母顺序qwq

利用ASCII码算出码值再转成字符即可 #include <bits/stdc.h> using namespace std;int main() {for(int i 1; i < 30; i){cout << char(ai) << ;} }结果&#xff1a; ps:大意了&#xff0c;本想用电脑目录&#xff0c;但没考虑到会有文件不存在导致缺…

[Unity Demo]从零开始制作空洞骑士Hollow Knight第二十集:制作专门渲染HUD的相机HUD Camera和画布HUD Canvas

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、制作HUD Camera以及让两个相机同时渲染屏幕二、制作HUD Canvas 1.制作法力条Soul Orb引入库2.制作生命条Health读入数据3.制作吉欧统计数Geo Counter4.制作…

python excel接口自动化测试框架!

今天采用Excel继续写一个接口自动化测试框架。 设计流程图 这张图是我的excel接口测试框架的一些设计思路。 首先读取excel文件&#xff0c;得到测试信息&#xff0c;然后通过封装的requests方法&#xff0c;用unittest进行测试。 其中&#xff0c;接口关联的参数通过正则进…

泷羽sec-linux

基础之linux 声明&#xff01; 学习视频来自B站up主 泷羽sec 有兴趣的师傅可以关注一下&#xff0c;如涉及侵权马上删除文章&#xff0c;笔记只是方便各位师傅的学习和探讨&#xff0c;文章所提到的网站以及内容&#xff0c;只做学习交流&#xff0c;其他均与本人以及泷羽sec团…

卷积神经网络学习记录

目录 神经网络基础定义&#xff1a; 基本组成部分 工作流程 卷积层&#xff08;卷积定义&#xff09;【CONV】&#xff1a; 卷积层&#xff08;Convolutional Layer&#xff09; 特征提取&#xff1a;卷积层的主要作用是通过卷积核&#xff08;或滤波器&#xff09;运算提…

计算机网络-GRE(通用路由封装协议)简介

昨天我们学习了VPN的基本概念&#xff0c;虚拟专用网络在当前企业总部与分支间广泛使用。常用的划分方法为基于协议层次有GRE VPN、IPSec VPN、L2TP VPN、PPTP VPN、SSL VPN等。其实我有考虑该怎么讲&#xff0c;因为在IP阶段好像虚拟专用网络讲得不深&#xff0c;在IE的阶段会…

SeggisV1 源码技术指导文档

软件下载地址&#xff1a; 百度网盘&#xff1a;链接:https://pan.baidu.com/s/1ZtwVcLsLypGo5lH6qR9oTw?pwd5856 问题咨询&#xff1a; https://github.com/YangJing524/Seggis

VSCode Terminal无法运行node以及node-gyp等指令

无法使用node指令&#xff0c;使用管理员权限启动VSCode即可&#xff0c;或者右键VSCode属性&#xff0c;修改兼容性中使用管理员权限打开。 运行node-gyp等指令出现因为在此系统上禁止运行脚本。有关详细信息&#xff0c;请参阅 https:/go.microsoft.com/fwlink/?LinkID1351…

前端全栈 === 快速入 门 Redis

目录 简介 通过 docker 的形式来跑&#xff1a; set、get 都挺简单&#xff1a; incr 是用于递增的&#xff1a; keys 来查询有哪些 key: redis insight GUI 工具。 list 类型 left push rpush lpop 和 rpop 自然是从左边和从右边删除数据。​编辑 如果想查看数据…

计算机网络socket编程(2)_UDP网络编程实现网络字典

个人主页&#xff1a;C忠实粉丝 欢迎 点赞&#x1f44d; 收藏✨ 留言✉ 加关注&#x1f493;本文由 C忠实粉丝 原创 计算机网络socket编程(2)_UDP网络编程实现网络字典 收录于专栏【计算机网络】 本专栏旨在分享学习计算机网络的一点学习笔记&#xff0c;欢迎大家在评论区交流讨…

简单链式表

# 完成双向循环链表的判空、尾插、遍历、尾删 class Node:def __init__(self, value):self.value valueself.next Noneself.prev None class Linklist:def __init__(self,nodeNone):self.head nodeself.size 0def is_empty(self):return self.size 0def add_tail(self,va…

流水线切分策略;通过自适应的重采样和重计算策略来优化计算资源和内存使用

目录 流水线切分策略 1,2,3,4,5指的计算任务(数据切分) 大方块代表GPU计算 黄色代表显存 通过自适应的重采样和重计算策略来优化计算资源和内存使用 一是自适应重计算(Adaptive Recomputation) 二是自适应划分(Adaptive Partitioning) 流水线切分策略 1,2,3,4,5指…

不只是请求和响应:使用Fiddler抓包URL和Method全指南(中)

欢迎浏览高耳机的博客 希望我们彼此都有更好的收获 感谢三连支持! 不只是请求和响应&#xff1a;使用Fiddler抓包HTTP协议全指南(上)-CSDN博客https://blog.csdn.net/Chunfeng6yugan/article/details/144005872?spm1001.2014.3001.5502 &#x1f649;在(上)篇博客中&#xf…