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.xdnf.cn/news/8927.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

Python数据分析-Netflix数据分析和可视化

一、研究背景 在当今时代&#xff0c;流媒体技术迅猛发展&#xff0c;如风暴般席卷全球娱乐产业&#xff0c;重塑了大众的娱乐消费模式。Netflix 在这一潮流中一马当先&#xff0c;成为全球首屈一指的在线流媒体平台。自 2007 年开启流媒体服务后&#xff0c;Netflix 就马不停…

数据集市是什么?有什么优势?

一、数据集市是什么&#xff1f; 1、数据集市的产生背景&#xff1a; 因为数据仓库的工作范围和成本比较巨大&#xff0c;技术部门必须对所有的以全企业的眼光对待任何一次决策分析&#xff0c;这样就变成了成本高、耗时高的大项目&#xff0c;而且这种集中式的数据处理方式往往…

Cross Modal Transformer: Towards Fast and Robust 3D Object Detection

代码地址 https://github.com/junjie18/CMT 1. 引言 在本文中&#xff0c;我们提出了Cross-Modal Transformer&#xff08;CMT&#xff09;&#xff0c;这是一种简单而有效的端到端管道&#xff0c;用于鲁棒的3D对象检测&#xff08;见图1&#xff08;c&#xff09;&#xf…

Oracle数据库 查看SQL执行计划的几种方法

前言 在日常的运维工作中&#xff0c;SQL优化是DBA的进阶技能&#xff0c;SQL优化的前提是要看SQL的执行计划是否正确&#xff0c;下面分享几种查看执行计划的方法&#xff0c;每一种方法都各有各的好处&#xff0c;可以根据特定场景选择某种方法。 一.使用AUTOTRACE查看执行…

简单介绍Nginx服务器的反向代理、负载均衡

欢迎来到“雪碧聊技术”CSDN博客&#xff01; 在这里&#xff0c;您将踏入一个专注于Java开发技术的知识殿堂。无论您是Java编程的初学者&#xff0c;还是具有一定经验的开发者&#xff0c;相信我的博客都能为您提供宝贵的学习资源和实用技巧。作为您的技术向导&#xff0c;我将…

域名+服务器+Nginx+宝塔使用SSL证书配置HTTPS

前言 在我的前面文章里&#xff0c;有写过一篇文章 linux服务器宝塔从头部署别人可访问的网站 在这篇文章&#xff0c;有教学怎么使用宝塔和买的服务器的公网IP&#xff0c;以及教怎么打包vue和springboot去部署不用域名的网站让别人访问 那么&#xff0c;这篇文章将在这个…

Chromium 中chrome.webRequest扩展接口定义c++

一、chrome.webRequest 注意 &#xff1a;从 Manifest V3 开始&#xff0c;"webRequestBlocking" 权限不再适用于大多数扩展程序。以 "declarativeNetRequest" 为例&#xff0c;它允许使用 declarativeNetRequest API。除了 "webRequestBlocking&quo…

.NET中通过C#实现Excel与DataTable的数据互转

在.NET框架中&#xff0c;使用C#进行Excel数据与DataTable之间的转换是数据分析、报表生成、数据迁移等操作中的常见需求。这一过程涉及到将Excel文件中的数据读取并加载至DataTable中&#xff0c;以便于利用.NET提供的丰富数据处理功能进行操作&#xff0c;同时也包括将DataTa…

多个NVR同时管理EasyNVR多品牌NVR管理工具/设备:IP常见问题解决方案

随着视频监控技术的不断发展&#xff0c;NVR&#xff08;网络视频录像机&#xff09;已经成为现代安防系统的重要组成部分。而为了更高效地管理多个品牌的NVR设备&#xff0c;EasyNVR这一多品牌NVR管理工具应运而生。然而&#xff0c;在实际使用过程中&#xff0c;尤其是在多个…

虚幻引擎 CEO 谈元宇宙:发展、策略与布局

在当今科技领域&#xff0c;元宇宙无疑是最热门的话题之一。Epic Games 首席执行官 Tim Sweeney 对元宇宙的未来发展充满信心&#xff0c;他认为开放元宇宙将融合娱乐、游戏和科技产业&#xff0c;带来一个光明的未来。本文将深入探讨采访中的关键内容&#xff0c;分析元宇宙的…

支付宝与华为终端联手,移动支付即将进入“碰时代”

大家好&#xff0c;我是小悟。 支付宝与华为终端强强联手&#xff0c;达成了战略合作&#xff01;这可不仅仅是个简单的合作哦&#xff0c;它预示着我们的移动支付方式即将迎来一场革命性的变革&#xff0c;正式进入“碰时代”&#xff01; 支付宝&#xff0c;作为全球领先的…

常用机器人算法原理介绍

一、引言 随着科技的不断发展&#xff0c;机器人技术在各个领域得到了广泛应用。机器人算法是机器人实现各种功能的核心&#xff0c;它决定了机器人的行为和性能。本文将介绍几种常用的机器人算法原理&#xff0c;包括路径规划算法、定位算法和运动控制算法。 二、路径规划算法…

【go从零单排】迭代器(Iterators)

&#x1f308;Don’t worry , just coding! 内耗与overthinking只会削弱你的精力&#xff0c;虚度你的光阴&#xff0c;每天迈出一小步&#xff0c;回头时发现已经走了很远。 &#x1f4d7;概念 在 Go 语言中&#xff0c;迭代器的实现通常不是通过语言内置的迭代器类型&#x…

Java 连接操作 MySQL 数据库(增删查改操作)

环境 MySQL 5.5 版本eclipseMySQL 连接驱动 mysql-connector-java-5.1.18-bin.jar mysql8.0之前的版本与之后的版本使用的jar包是不同的&#xff0c;在使用时也有一定的区别。这里&#xff0c;我的 MySQL 版本为 5.5。 准备工作 将 jar 包添加到项目中&#xff0c;右键项目&a…

STL---迭代器

本文来源&#xff1a;《C语言程序设计》第10章 理解迭代器对于理解STL框架并掌握STL的使用至关重要。 迭代器是泛化的指针&#xff0c;STL算法利用迭代器对存储在容器中的元素序列进行遍历&#xff0c;迭代器提供了访问容器中每个元素的方法。 虽然指针也是一种迭代器&#…

TSMI252012PMX-3R3MT功率电感详细解析

TSMI252012PMX-3R3MT功率电感详细解析 一、引言 在现代电子设备的不断小型化和高性能化的趋势下&#xff0c;功率电感作为电路中的关键元件&#xff0c;其性能的好坏直接影响到整个电路的稳定性和效率。TSMI252012PMX-3R3MT作为深圳市时源芯微科技有限公司&#xff08;TimeSo…

Ubuntu22.04安装DataEase

看到DataEase的驾驶舱&#xff0c;感觉比PowerBI要好用一点&#xff0c;于是搭建起来玩玩。Dataease推荐的操作系统是Ubuntu22.04/Centos 7。 下载了Ubuntu22.04和DataEase 最新版本的离线安装包 一.安装ubuntu22.04 在安装的时候&#xff0c;没有顺手设置IP地址信息&#xff…

OpenEuler 下 Docker 安装、配置与测试实例

文章目录 前言1. 环境准备2. 下载 Docker3.配置服务文件4.配置加速器加速下载docker镜像5. 验证 Docker 安装 前言 Docker 安装大致分为包管理器安装、脚本安装、离线手动安装、容器编排工具安装、桌面版安装等&#xff0c;每种安装各有特点&#xff0c;但涉及知识面不少&…

wordpress实用功能A5资源网同款 隐藏下载框 支付框 需要登录才能查看隐藏的内容

实用功能 隐藏下载框 支付框 需要登录才能查看隐藏的内容, 个人网站防天朝申查实测有效 。 登录前&#xff0c;未登录&#xff1a; 登录后&#xff0c;已登录&#xff1a; 功能说明 该代码段的主要功能是隐藏支付框并为未 登录用户显示一条提示信息&#xff0c;告知他们需要…

C 语言学习-05【数组】

1、一维数组元素的操作 输入一个数&#xff0c;按原来排序的规律将它插入到一个一排列好的数组中&#xff1a; #include <stdio.h>int main() {int i, data, a[10] {2, 3, 6, 9, 11, 12, 14, 17, 19};printf("Primitive series: \n");for (i 0; i < 9; i)…