C语言学习--const修饰符的作用,如何强制修改const定义的对象内容

C语言学习--const修饰符的作用,如何强制修改const定义的对象内容

  • const 在 C 语言中的作用及应用场景
    • 1.修饰变量
      • 应用场景:
    • 2. 修饰指针
      • 应用场景:
    • 修饰函数
      • 应用场景:
    • 特殊用法,修饰寄存器
      • 应用场景:
  • 如何在运行过程中修改 const 变量参数
      • 示例实例
    • 2.1 通过指针强制转换修改 const 修饰的成员
      • 示例:修改 const uint32_t capacity
      • 修改 uint32_t *const start_ptr 指针常量的值
    • 2.2 利用union联合体修改 const 修饰的成员
      • 修改 const 成员示例
      • 使用示例
  • 注意事项

const 在 C 语言中的作用及应用场景

const 是 C 语言中的一个关键字,用于定义不可修改的变量。通过在变量声明中使用 const,可以指示编译器在代码中禁止对该变量进行修改,增加代码的安全性和可读性。以下是几个典型的应用场景:

1.修饰变量

使用 const 修饰变量可以定义不可变的常量,通常用来保护数据避免在后续的代码中被意外修改。

c
#include <stdio.h>int main() {const int num = 10; // 定义不可修改的整型常量// num = 20; // 错误:不能修改 const 修饰的变量printf("num = %d\n", num);return 0;
}

应用场景:

  • 防止程序员无意中修改数据。
  • 确保变量在整个程序执行过程中保持不变。

2. 修饰指针

const 修饰指针时,有以下几种常见用法:

  • 常量指针 (const int *ptr): 指针所指向的数据不可修改,但指针自身可以修改。
const int value = 10;
const int *ptr = &value;
// *ptr = 20; // 错误:不能修改 ptr 所指向的数据
ptr = NULL;   // 合法:可以改变 ptr 本身的指向
  • 指针常量 (int * const ptr): 指针自身不可修改,但指向的数据可以修改。
int value = 10;
int * const ptr = &value;
*ptr = 20;    // 合法:可以修改 ptr 所指向的数据
// ptr = NULL; // 错误:不能修改指针本身的指向
  • 常量指针常量 (const int * const ptr): 指针本身和它所指向的数据都不可修改。
const int value = 10;
const int * const ptr = &value;
// *ptr = 20; // 错误:不能修改 ptr 所指向的数据
// ptr = NULL; // 错误:不能修改指针本身的指向

应用场景:

  • 防止指针指向的数据或指针本身被无意中修改,提高代码的健壮性。

修饰函数

使用 const 修饰函数参数,特别是在函数参数为指针时,可以有效防止函数内部修改该参数指向的内容。

void print_array(const int *array, size_t size) {for (size_t i = 0; i < size; i++) {printf("%d ", array[i]);// array[i] = 0; // 错误:不能修改 const 修饰的内容}
}

应用场景:

防止函数修改输入参数的值,特别是在需要保护传入数组或结构体的情况下。

特殊用法,修饰寄存器

在嵌入式编程中,寄存器通常是只读的,使用 const 修饰寄存器可以表明该寄存器值不会被程序更改,仅用于读取操作。

volatile const uint32_t * const REGISTER_STATUS = (uint32_t *)0x40001000;

应用场景:

  • 使用 const 修饰硬件寄存器可以防止无意写入寄存器内容。

如何在运行过程中修改 const 变量参数

在正常情况下,const 变量在编译时被定义为不可修改。但是在一些特殊情况下(例如硬件寄存器或调试场景),我们可能需要修改 const 变量的内容。下面介绍一些绕过 const 限制的方法。

以下是一个结构体定义,包含不同形式的 const 修饰变量和指针:


```c
```c
#include <stdio.h>
#include <stdint.h>
#include <stdbool.h>// 示例结构体,包含不同的 const 成员
typedef struct {const uint32_t capacity;                 // 常量变量,不能修改uint32_t *const start_ptr;               // 指针常量,指针本身不能改变const uint32_t *current_ptr;             // 常量指针,指向的数据不能修改uint32_t *next_ptr;                      // 普通指针,无 const 限制
} RingBuffer;
  • 对于上面的数据结构,我们希望修改capacity以及s指针常量tart_ptr的值,常量指针和普通指针,都能直接修改指针指向的对象,达到修改的目的。

示例实例

创建 RingBuffer 结构体的实例,其中包含不同的 const 修饰成员:

uint32_t buffer[100];
RingBuffer ring_buffer = {.capacity = 100,.start_ptr = buffer,.current_ptr = buffer,.next_ptr = buffer
};

2.1 通过指针强制转换修改 const 修饰的成员

对于 const 修饰的普通变量或指针,我们可以通过强制转换来进行修改。这种方式会绕过编译器的 const 限制。

示例:修改 const uint32_t capacity

capacity 是一个 const 修饰的变量。我们可以通过强制类型转换将其转换为 uint32_t * 类型来绕过 const 限制。

#include <stdio.h>void modify_capacity(RingBuffer *rb, uint32_t new_capacity) {(*(uint32_t *)&rb->capacity) = new_capacity;  // 强制类型转换绕过 const 限制
}int main() {printf("Original capacity: %u\n", ring_buffer.capacity);modify_capacity(&ring_buffer, 200);printf("Modified capacity: %u\n", ring_buffer.capacity);  // 输出:Modified capacity: 200return 0;
}

修改 uint32_t *const start_ptr 指针常量的值

start_ptr 是一个指针常量,表示指针本身不能修改,可以通过二级指针强制转换的方式,修改start_ptr的指向

uint32_t new_buffer[100];
*((uint32_t **)&ring_buffer ->start_ptr) = &new_buffer;  // 强制类型转换绕过 const 限制

2.2 利用union联合体修改 const 修饰的成员

利用联合体(union)来修改 const 属性的原理在于联合体的所有成员共享相同的内存空间。通过访问联合体的非 const 成员,我们可以绕过 const 修饰符,直接修改 const 数据。这种做法常用于底层编程、嵌入式系统开发或特定调试需求中。

在联合体中,我们定义了 RingBuffer 结构体和一个原始字节数组 raw_data,两者共享同一块内存空间。通过 raw_data 访问数据,我们可以绕过 const 限制修改 RingBuffer 中的 const 成员。

typedef union {RingBuffer ring_buffer;                 // 原始的 RingBuffer 结构体uint8_t raw_data[sizeof(RingBuffer)];   // 访问相同内存的字节数组
} RingBufferUnion;

修改 const 成员示例

我们通过 RingBufferUnion 修改 RingBuffer 中的 const 成员 capacity 和 start_ptr。首先,将 RingBuffer 初始化为只读内容,然后通过联合体的 raw_data 字段修改 const 成员。

void modify_const_members_via_union(RingBufferUnion *rb_union, uint32_t new_capacity, uint32_t *new_start_ptr) {// 强制将 raw_data 的前几个字节转换为 uint32_t,修改 capacity*((uint32_t *)rb_union->raw_data) = new_capacity;// 修改 start_ptr 的值*((const uint32_t **)(rb_union->raw_data + sizeof(uint32_t))) = new_start_ptr;
}

使用示例

我们定义一个 RingBuffer 实例并尝试修改其 const 成员。

int main() {uint32_t buffer[100];RingBufferUnion rb_union = {.ring_buffer = {.capacity = 100,               // 初始化容量.start_ptr = buffer,.end_ptr = buffer + 100,.current_ptr = buffer}};printf("Original capacity: %u\n", rb_union.ring_buffer.capacity);printf("Original start_ptr: %p\n", (void *)rb_union.ring_buffer.start_ptr);// 通过联合体修改 const 成员uint32_t new_capacity = 200;uint32_t new_start_buffer[100];modify_const_members_via_union(&rb_union, new_capacity, new_start_buffer);printf("Modified capacity: %u\n", rb_union.ring_buffer.capacity);printf("Modified start_ptr: %p\n", (void *)rb_union.ring_buffer.start_ptr);return 0;
}

注意事项

  • 破坏 const 语义:此方法强制修改指针常量的指向,违背了 const 的语义,需在应用场景中确保安全。
  • 底层访问风险:联合体强制访问原始数据结构有一定的风险,建议仅在底层代码或调试环境中使用。

通过此方法,可以在不改变结构体定义的情况下强制修改指针常量的指向。不过这种操作在生产环境中应慎用。

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

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

相关文章

C++:map 和 set 的使用

前言 平衡二叉搜索树 ( AVL树 ) 由于二叉搜索树在特殊情况下&#xff0c;其增删查的效率会降低到 O ( N )&#xff0c;因此对二叉搜索树进行改良&#xff0c;通过旋转等方式将其转换为一个左右均衡的二叉树&#xff0c;这样的树就称为平衡二叉搜索树&#xff0c;又称 AVL树。…

Vue 自定义icon组件封装SVG图标

通过自定义子组件CustomIcon.vue使用SVG图标&#xff0c;相比iconfont下载文件、重新替换更节省时间。 子组件包括&#xff1a; 1. Icons.vue 存放所有SVG图标的path 2. CustomIcon.vue 通过icon的id索引对应的图标 使用的时候需要将 <Icons></Icons> 引到使用的…

面相小白的php反序列化漏洞原理剖析

前言 欢迎来到我的博客 个人主页:北岭敲键盘的荒漠猫-CSDN博客 本文整理反序列化漏洞的一些成因原理 建议学习反序列化之前 先对php基础语法与面向对象有个大体的了解 (我觉得我整理的比较细致&#xff0c;了解这俩是个啥就行) 漏洞实战情况 这个漏洞黑盒几乎不会被发现&am…

ReactPress:深入解析技术方案设计与源码

ReactPress Github项目地址&#xff1a;https://github.com/fecommunity/reactpress 欢迎提出宝贵的建议&#xff0c;欢迎一起共建&#xff0c;感谢Star。 ReactPress是一个基于React框架开发的开源发布平台&#xff0c;它不仅仅是一个简单的博客系统&#xff0c;更是一个功能全…

canal1.1.7使用canal-adapter进行mysql同步数据

重要的事情说前面&#xff0c;canal1.1.8需要jdk11以上&#xff0c;大家自行选择&#xff0c;我这由于项目原因只能使用1.1.7兼容版的 文章参考地址&#xff1a; canal 使用详解_canal使用-CSDN博客 使用canal.deployer-1.1.7和canal.adapter-1.1.7实现mysql数据同步_mysql更…

SpringBoot之定时任务

1. 前言 本篇博客是个人的经验之谈&#xff0c;不是普适的解决方案。阅读本篇博客的朋友&#xff0c;可以参考这里的写法&#xff0c;如有不同的见解和想法&#xff0c;欢迎评论区交流。如果此篇博客对你有帮助&#xff0c;感谢点个赞~ 2. 场景 我们讨论在单体项目&#xff0c…

绿色能源发展关键:优化风电运维体系

根据QYResearch调研团队最新发布的《全球风电运维市场报告2023-2029》显示&#xff0c;预计到2029年&#xff0c;全球风电运维市场的规模将攀升至307.8亿美元&#xff0c;并且在接下来的几年里&#xff0c;其年复合增长率&#xff08;CAGR&#xff09;将达到12.5%。 上述图表及…

前端 Canvas 绘画 总结

目录 一、使用案例 1、基础使用案例 2、基本案例改为直接JS实现 二、相关资料 1、API教程文档 2、炫酷案例 一、使用案例 1、基础使用案例 使用Canvas的基本步骤&#xff1a; 1、需要一个canvas标签 2、需要获取 画笔 对象 3、使用canvas提供的api进行绘图 <!--…

力扣排序455题(分发饼干)

假设你是一位很棒的家长&#xff0c;想要给你的孩子们一些小饼干。 但是&#xff0c;每个孩子最多只能给一块饼干。 对每个孩子 i&#xff0c;都有一个胃口值 g[i],这是能 让孩子们满足胃口的饼干的最小尺寸;并且每块饼 干j&#xff0c;都有一个尺寸 s[j]。如果 s[j]> g[i]&…

C语言 | Leetcode C语言题解之第537题复数乘法

题目&#xff1a; 题解&#xff1a; bool parseComplexNumber(const char * num, int * real, int * image) {char *token strtok(num, "");*real atoi(token);token strtok(NULL, "i");*image atoi(token);return true; };char * complexNumberMulti…

Android使用scheme方式唤醒处于后台时的App场景

场景&#xff1a;甲App唤醒处于后台时的乙App的目标界面Activity&#xff0c;且乙App的目标界面Activity处于最上层&#xff0c;即已经打开状态&#xff0c;要求甲App使用scheme唤醒乙App时&#xff0c;达到跟从桌面icon拉起App效果一致&#xff0c;不能出现只拉起了乙App的目标…

如何对接低价折扣相对稳定电影票渠道?

对接低价折扣电影票渠道需要经过一系列步骤&#xff0c;以确保能够为用户提供优惠且可靠的购票体验。以下是一个基本的对接流程&#xff1a; 1.市场调研&#xff1a; 调研市场上的电影票销售渠道&#xff0c;了解主要的电影票批发商和分销商。分析竞争对手的折扣电影票服务&a…

【上云拼团Go】如何在腾讯云双十一活动中省钱

1. 前言 双十一已经成为了全球最大的购物狂欢节&#xff0c;除了电商平台的优惠&#xff0c;云计算服务商也纷纷在这个期间推出了诱人的促销活动。腾讯云作为中国云计算的领军企业之一&#xff0c;每年双十一的活动都吸引了大量开发者、企业和个人用户参与。那么&#xff0c;在…

新能源企业在精益变革过程中可能会遇到哪些困难?

在绿色转型的浪潮中&#xff0c;新能源企业作为推动社会可持续发展的先锋力量&#xff0c;正加速迈向精益化管理的新阶段。然而&#xff0c;这条变革之路并非坦途&#xff0c;而是布满了未知与挑战。本文&#xff0c;天行健王春城老师将深入探讨新能源企业在精益变革过程中可能…

Maven的安装配置

文章目录 一、MVN 的下载二、配置maven2.1、更改maven/conf/settings.xml配置2.2、配置环境变量一、MVN 的下载 还是那句话,要去就去官网或者github,别的地方不要去下载。我们下载binaries/ 目录下的 cd /opt/server wget https://downloads.apache.org/maven/maven-3/3.9.6/…

OpenCV视觉分析之目标跟踪(10)估计两个点集之间的刚性变换函数estimateRigidTransform的使用

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 计算两个2D点集之间的最优仿射变换 estimateRigidTransform 是 OpenCV 中的一个函数&#xff0c;用于估计两个点集之间的刚性变换&#xff08;即…

块存储、文件存储和对象存储详细介绍

块存储、文件存储和对象存储介绍 块存储&#xff1a;像跑车&#xff0c;因为它们都能提供快速的响应和高性能&#xff0c;适合需要即时数据访问的场景&#xff0c;比如数据库和虚拟化技术。 文件存储&#xff1a;像货车&#xff0c;因为它们都能承载大量货物&#xff08;文件&…

A019基于SpringBoot的校园闲置物品交易系统

&#x1f64a;作者简介&#xff1a;在校研究生&#xff0c;拥有计算机专业的研究生开发团队&#xff0c;分享技术代码帮助学生学习&#xff0c;独立完成自己的网站项目。 代码可以查看文章末尾⬇️联系方式获取&#xff0c;记得注明来意哦~&#x1f339; 赠送计算机毕业设计600…

基于YOLO11/v10/v8/v5深度学习的煤矿传送带异物检测系统设计与实现【python源码+Pyqt5界面+数据集+训练代码】

《------往期经典推荐------》 一、AI应用软件开发实战专栏【链接】 项目名称项目名称1.【人脸识别与管理系统开发】2.【车牌识别与自动收费管理系统开发】3.【手势识别系统开发】4.【人脸面部活体检测系统开发】5.【图片风格快速迁移软件开发】6.【人脸表表情识别系统】7.【…

explain执行计划分析 ref_

这里写目录标题 什么是ExplainExplain命令扩展explain extendedexplain partitions 两点重要提示本文示例使用的数据库表Explain命令(关键字)explain简单示例explain结果列说明【id列】【select_type列】【table列】【type列】 【possible_keys列】【key列】【key_len列】【ref…