嵌入式之指针
在嵌入式系统中指针是一种非常重要的概念。它们用于直接访问内存地址,能够提高程序的灵活性和效率。
一、基本概念
1. 指针的基本概念
- 定义:指针是一个变量,其值为另一个变量的地址。通过指针,可以间接访问和修改该变量的值。
- 声明:在C语言中,指针的声明格式为
类型 *指针名,例如int *p表示p是一个指向整数的指针。
2. 指针的使用
- 动态内存分配:在嵌入式系统中,动态内存分配(如使用
malloc)可以根据需要分配内存,但要注意内存的管理,避免内存泄漏。 - 数组和指针:数组名在大多数情况下可以看作是指向数组首元素的指针。通过指针可以方便地遍历数组。
- 函数参数:使用指针作为函数参数,可以实现对原始数据的修改,避免数据的复制,提高效率。
3. 指针的类型
- 普通指针:指向基本数据类型(如
int、char等)。 - 结构体指针:指向结构体类型,可以方便地访问结构体中的各个成员。
- 函数指针:指向函数的指针,可以实现回调机制,增加程序的灵活性。
4. 指针的注意事项
- 指针的初始化:使用指针前应确保其已被初始化,避免出现野指针(即指向未知地址的指针)。
- 内存管理:在嵌入式系统中,内存资源往往有限,因此要谨慎使用动态内存,尽量避免频繁分配和释放内存。
- 越界访问:使用指针时要注意数组的边界,避免越界访问,导致未定义行为。
二、指针类型
1. 普通指针
普通指针用于指向基本数据类型(int、char)。它们可以用于动态内存管理、数组操作和函数参数传递等场景。
例子:普通指针的基本用法
#include <stdio.h>int main() {int a = 10; // 定义一个整数int *p = &a; // 声明一个指针并初始化为 a 的地址printf("Value of a: %d\n", a); // 输出 a 的值printf("Value via pointer: %d\n", *p); // 通过指针访问 a 的值*p = 20; // 通过指针修改 a 的值printf("New value of a: %d\n", a); // 输出修改后的 a 的值return 0;
}
应用场景
- 动态内存分配:使用
malloc和free进行动态内存管理。 - 数组操作:通过指针遍历和操作数组元素。
- 函数参数传递:避免大数据结构的复制,提高性能。
2. 结构体指针
结构体指针用于指向结构体类型,可以方便地访问结构体中的各个成员。
例子:结构体指针的用法
#include <stdio.h>
#include <string.h>typedef struct {char name[20];int age;
} Person;int main() {Person person; // 定义一个结构体变量Person *pPerson = &person; // 声明一个指向结构体的指针// 使用结构体指针访问和修改成员strcpy(pPerson->name, "Alice");pPerson->age = 30;printf("Name: %s, Age: %d\n", pPerson->name, pPerson->age);return 0;
}
应用场景
- 数据组织:使用结构体指针管理复杂数据结构(如链表、树等)。
- 函数参数传递:通过结构体指针传递大型结构体,避免复制,提高效率。
3. 函数指针
函数指针是指向函数的指针,可以用来实现回调机制、事件处理等。
例子:函数指针的用法
#include <stdio.h>// 定义一个函数类型
typedef void (*Operation)(int, int);// 加法函数
void add(int a, int b) {printf("Sum: %d\n", a + b);
}// 减法函数
void subtract(int a, int b) {printf("Difference: %d\n", a - b);
}// 执行操作的函数
void executeOperation(Operation op, int x, int y) {op(x, y); // 调用传入的函数指针
}int main() {// 声明函数指针并指向加法和减法函数Operation op1 = add;Operation op2 = subtract;// 执行操作executeOperation(op1, 10, 5); // 调用加法executeOperation(op2, 10, 5); // 调用减法return 0;
}
应用场景
- 回调函数:在事件驱动编程中,使用函数指针作为回调。
- 策略模式:根据不同的条件选择不同的函数执行。
- 状态机:通过函数指针实现不同状态下的行为。
三、易混辨别
1. 常量指针与指针常量
1.1 常量指针
常量指针是指指针所指向的内容是常量,不能通过该指针修改其指向的值。常量指针的语法是 const 类型 *指针名。
#include <stdio.h>void printValue(const int *ptr) {// ptr指向的值是常量,不能通过ptr修改printf("Value: %d\n", *ptr);// *ptr = 20; // 错误:不能修改常量
}int main() {int a = 10;const int *p = &a; // p是一个常量指针,指向整数printValue(p); // 输出值// *p = 15; // 错误:不能通过常量指针修改值return 0;
}
- 保护数据:常量指针用于保护指向的数据不被修改,适合于需要只读访问的场景。
- 函数参数:在函数参数中使用常量指针,可以避免不小心修改传入的变量。
1.2 指针常量
指针常量是指指针本身的值是常量,不能改变指向的地址。指针常量的语法是 类型 * const 指针名。
#include <stdio.h>int main() {int a = 10;int b = 20;int * const p = &a; // p是一个指针常量,指向整数printf("Value pointed by p: %d\n", *p); // 输出a的值*p = 15; // 可以通过指针修改指向的值printf("New value of a: %d\n", a); // 输出修改后的a的值// p = &b; // 错误:不能改变指针常量的指向return 0;
}
- 固定指针:指针常量用于确保指针在其生命周期内指向同一地址,适合需要固定指向某个数据的场景。
- 避免意外更改:在复杂的函数中,确保指针不被意外更改,增加代码的可读性和安全性。
1.3 常量指针与指针常量的区别
| 特性 | 常量指针 (const T*) | 指针常量 (T* const) |
|---|---|---|
| 修改指向的内容 | 不可修改 | 可修改 |
| 修改指针的地址 | 可修改 | 不可修改 |
| 语法 | const int *p | int * const p |
- 常量指针:指向的内容不可修改,但指针本身可以指向其他地址。
- 指针常量:指针本身不可修改,但可以通过指针修改指向的内容。
- 结合使用:可以创建一个指向常量的指针常量,既不能修改指向的内容,也不能改变指针的指向。
1.4 结合使用
常量指针和指针常量可以结合使用,形成一个指向常量的指针常量,语法为 const 类型 * const 指针名。
#include <stdio.h>int main() {int a = 10;const int * const p = &a; // p是一个指向常量的指针常量printf("Value pointed by p: %d\n", *p); // 输出a的值// *p = 15; // 错误:不能通过常量指针修改值// p = &b; // 错误:不能改变指针常量的指向return 0;
}
2. 函数指针与指针函数
2.1 函数指针
概念
- 函数指针是一个变量,它存储的是一个函数的地址。通过函数指针,可以调用指向的函数,就像直接调用函数一样。
#include <iostream>// 定义两个简单的函数
int add(int a, int b) {return a + b;
}int subtract(int a, int b) {return a - b;
}// 定义一个函数指针类型
typedef int (*Operation)(int, int);void performOperation(Operation op, int x, int y) {std::cout << "Result: " << op(x, y) << std::endl;
}int main() {// 使用函数指针调用不同的函数performOperation(add, 5, 3); // 输出: Result: 8performOperation(subtract, 5, 3); // 输出: Result: 2return 0;
}
用途
- 动态函数调用:可以在运行时决定调用哪个函数。
- 回调机制:将函数指针作为参数传递给另一个函数,实现回调。
- 实现策略模式:通过函数指针选择不同的算法或策略。
2.2 指针函数
概念
- 指针函数是一个返回指针的函数。函数的返回类型是一个指针类型。
例子
#include <iostream>// 返回一个指向数组中最大元素的指针
int* findMax(int* arr, int size) {if (size <= 0) return nullptr;int* max = &arr[0];for (int i = 1; i < size; ++i) {if (arr[i] > *max) {max = &arr[i];}}return max;
}int main() {int numbers[] = {1, 5, 3, 9, 2};int size = sizeof(numbers) / sizeof(numbers[0]);int* maxPtr = findMax(numbers, size);if (maxPtr != nullptr) {std::cout << "Max value: " << *maxPtr << std::endl; // 输出: Max value: 9}return 0;
}
用途
- 返回动态内存地址:指针函数常用于返回动态分配的内存地址。
- 返回复杂数据结构的地址:例如返回数组中某个元素的地址。
- 链表或树结构的导航:在数据结构中,指针函数可以用于遍历或查找节点。
总结如下:
| 特性 | 函数指针 | 指针函数 |
|---|---|---|
| 定义 | 存储函数地址的指针变量。 | 返回指针类型的函数。 |
| 语法 | return_type (*pointer_name)(parameter_list); | return_type* function_name(parameter_list); |
| 用途 | 用于动态调用函数、实现回调机制、策略模式等。 | 用于返回动态内存地址或复杂数据结构的地址。 |
| 应用场景 | - 回调函数实现 - 动态函数选择 - 策略模式实现 | - 返回数组或对象的地址 - 链表或树节点的导航 |
| 调用方式 | 通过解引用指针来调用所指向的函数。 | 直接调用函数以获得返回的指针。 |
| 示例 | int (*funcPtr)(int, int) = add; | int* findMax(int* arr, int size); |
| 灵活性 | 高,因为可以在运行时选择不同的函数进行调用。 | 提供了一种返回复杂数据结构或动态内存的方式。 |
相关文章:
嵌入式之指针
在嵌入式系统中指针是一种非常重要的概念。它们用于直接访问内存地址,能够提高程序的灵活性和效率。 一、基本概念 1. 指针的基本概念 定义:指针是一个变量,其值为另一个变量的地址。通过指针,可以间接访问和修改该变量的值。声…...
网络安全研究
1.1 网络安全面临的威胁 网络安全面临的威胁呈现出多样化和复杂化的趋势,给个人、企业和国家的安全带来了严峻挑战。以下是当前网络安全面临的主要威胁: 1.1.1 数据泄露风险 数据泄露是当前网络安全的重大威胁之一。根据国家互联网应急中心发布的《20…...
Git入门:数据模型 to 底层原理
版本控制系统(VCS)是软件开发中不可或缺的工具,而Git作为现代版本控制的事实标准,其底层设计远比表面命令更加优雅。本文将从数据模型的角度,揭示Git的核心工作原理。 Git的核心概念 1. 快照(Snapshot&am…...
openharmony中hdf框架的驱动消息机制的实现原理
openharmony中hdf框架的驱动消息机制的实现原理 在分析hdf框架时发现绕来绕去的,整体梳理画了一遍流程图,发现还是有点模糊甚至不清楚如何使用的,详细的每个点都去剖析细节又过于消耗时间,所以有时间便从功能应用的角度一块块的去…...
HTTP SSE 实现
参考: SSE协议 SSE技术详解:使用 HTTP 做服务端数据推送应用的技术 一句概扩 SSE可理解为:服务端和客户端建立连接之后双方均保持连接,但仅支持服务端向客户端推送数据。推送完毕之后关闭连接,无状态行。 下面是基于…...
二分图检测算法以及最大匹配算法(C++)
上一节我们学习了有向图中的最大连通分量. 本节我们来学习二分图. 二分图是一种特殊的图结构, 能够帮助我们高效地解决这些匹配和分配问题. 本文将带你了解二分图的基本概念, 判定方法, 最大匹配算法以及实际应用场景. 环境要求 本文所用样例在Windows 11以及Ubuntu 24.04上面…...
Keepalive基础
一。简介和功能 vrrp协议的软件实现,原生设计目的是为了高可用ipvs服务 功能: 1.基于vrrp协议完成地址流动 2.为vip地址所在的节点生成ipvs规则(在配置文件中预先定义) 3.为ipvs集群的各RS做健康状况检测 4.基于脚本调用接口…...
计算机毕业设计SpringBoot+Vue.jst0图书馆管理系统(源码+LW文档+PPT+讲解)
温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 温馨提示:文末有 CSDN 平台官方提供的学长联系方式的名片! 作者简介:Java领…...
【Java消息队列】应对消息丢失、重复、顺序与积压的全面策略
应对消息丢失、重复、顺序与积压的全面策略 引言kafka消息丢失生产者消费者重复消费顺序消费消息积压生产者消费者其他RabbitMQ消息丢失生产者事务机制,保证生产者发送消息到 RabbitMQ Server发送方确认机制,保证消息能从交换机路由到指定队列保证消息在 RabbitMQ Server 中的…...
AI大模型学习(三): LangChain(二)
Langchain构建聊天机器人 安装依赖 pip install langchain_community Chat History:它允许聊天机器人"记住"过去的互动,并在回应后续问题时考虑他们 代码 # 创建模型 from langchain_core.messages import HumanMessage from langchain_core.prompts import ChatP…...
apply的用法
apply 是一个在编程语言中常见的函数,它在不同的上下文和语言中有不同的用途。以下是 apply 在常见编程语言中的几种常见用法: 1. Python 中的 apply 方法 在 Python 中,apply 主要用于 pandas 库中的 DataFrame 或 Series 对象,…...
【论文解读】TransMLA: Multi-Head Latent Attention Is All You Need
论文链接 1. 论文背景与问题动机 现代大规模语言模型(LLM)在推理时往往遇到通信瓶颈,主要原因在于自注意力机制中需要缓存大量的 Key-Value(KV)对。例如,对于 LLaMA‑65B 这种模型,即使采用 8…...
CentOS 下安装和配置 HTTPD 服务的详细指南
CentOS 下安装和配置 HTTPD 服务的详细指南 CentOS 下安装和配置 HTTPD 服务的详细指南1. 环境准备2. 安装 HTTPD 服务2.1 更新系统2.2 安装 HTTPD2.3 启动 HTTPD 服务2.4 检查 HTTPD 服务状态 3. 配置防火墙3.1 开放 HTTP 和 HTTPS 端口3.2 验证防火墙规则 4. 配置 HTTPD4.1 主…...
VUE3中子组件改变父组件传过来的值(props)的方法和使用场景详解
在 Vue 3 中,子组件改变父组件传过来的值(props)的方法主要有以下几种:通过事件发射、使用 v-model、模拟 .sync 修饰符的功能(Vue 3 中已移除),以及使用 ref 或 reactive。下面我将结合代码示例…...
登录-06.JWT令牌-生成和校验
一.JWT令牌的生成和校验 JWT令牌生成 想要生成JWT令牌,那么就要首先引入JWT令牌的相关依赖, <dependency><groupId>io.jsonwebtoken</groupId><artifactId>jjwt-api</artifactId><version>0.11.2</version>…...
【Git】多人协作
文章目录 完成准备工作多人协作场景一场景二远程分支删除后,本地 git branch -a 依然能看到的解决办法 完成准备工作 在之前,我们所完成的工作如下: 基本完成 Git 的所有本地库的相关操作,git基本操作,分支理解&#…...
Python爬虫-破解字体加密技术
前言 本文是该专栏的第77篇,后面会持续分享python爬虫干货知识,记得关注。 字体加密是一种常见的反爬虫技术,通过自定义字体文件和字符映射来保护网页内容,防止爬虫直接获取文本信息。 在文章《Python爬虫-猫眼电影的影院数据》中,笔者有详细介绍过猫眼的相关数据采集。…...
邮件安全之发件人伪造
电子邮件工作原理 电子邮件传输过程中主要涉及到SMTP、IMAP、POP3三种协议,具体功能如下: SMTP:全称Simple Mail Transfer Protocol,即简单邮件传输协议,主要用于发送邮件,使用端口号25。 IMAP:全称Internet Mail Acce…...
git 常用功能
以下是 Git 的常用功能及其命令: 初始化仓库 git init在当前目录初始化一个新的 Git 仓库。 克隆仓库 git clone <仓库地址>将远程仓库克隆到本地。 查看状态 git status查看工作区和暂存区的状态。 添加文件到暂存区 git add <文件名>将文件添…...
【llm落地】从零到一,用DeepSeek打造智能BI工具:自然语言驱动数据洞察
在数据驱动的时代,商业智能 (BI) 工具已经成为企业决策的关键。然而,传统的 BI 工具往往操作复杂,需要专业技能才能驾驭。想象一下,如果用户只需要用 自然语言 就能轻松查询数据、获取分析结果甚至生成可视化图表,那将会多么高效和便捷! 本文将带你踏上从零到一构建智能…...
请谈谈 Vue 中的 key 属性的重要性,如何确保列表项的唯一标识?
1. Key属性的核心作用(附代码对比) // 错误示例:未使用key的列表渲染 <template><ul><li v-for"item in items">{{ item.text }}</li></ul> </template>// 正确示例:使用唯一key的…...
使用 AIStor 和 OpenSearch 增强搜索功能
在这篇文章中,我们将探讨搜索,特别是 OpenSearch 如何帮助我们识别模式或查看不断增长的数据中的趋势。例如,如果您正在查看运营数据,如果您的服务似乎是随机的,那么您需要尽可能回溯以识别模式并找出原因。这不仅适用…...
Node.js中如何修改全局变量的几种方式
Node.js中如何修改全局变量。我需要先理解他们的需求。可能他们是在开发过程中遇到了需要跨模块共享数据的情况,或者想要配置一些全局可访问的设置。不过,使用全局变量可能存在一些问题,比如命名冲突、难以维护和测试困难,所以我得…...
基于Python和Neo4j开发的医疗辅助诊断系统的详细实现步骤和代码示例
以下是一个基于Python和Neo4j开发的医疗辅助诊断系统的详细实现步骤和代码示例。 1. 环境准备 首先,确保你已经安装了必要的库。可以使用以下命令进行安装: pip install py2neo2. Neo4j数据库初始化 在Neo4j中创建一个新的数据库,并启动N…...
第9章:LangChain结构化输出-示例2(数字提取服务)
如何使用LangChain4j框架创建和使用多种AI服务。它通过定义接口和注解,将自然语言处理任务(如情感分析、数字提取、日期提取、POJO提取等)封装为服务,并通过LangChain4j的AiServices动态生成这些服务的实现。 本章主要讲述基于Lan…...
【LLM】R1复现项目(SimpleRL、OpenR1、LogitRL、TinyZero)持续更新
note (1)未来的工作需亟待解决: 支持大规模 RL 训练(PPO、GRPO 等)的开源基础框架用于稳定训练的 GRPO 训练超参的自动化调优RL 训练数据的配比(难度、领域、任务等)基于 Instruct 模型训练 R…...
买股票的最佳时机 - 2
买卖股票的最佳时机 III 题目描述: 提示: 1 < prices.length < 1050 < prices[i] < 105 分析过程: 写动态规划,我们需要考虑一下问题: 定义状态状态转移方程初始条件 遍历顺序 4种状态: …...
Python基于flask的智慧交通可视化,大数据智慧交通数据可视化系统
博主介绍:✌程序员徐师兄、8年大厂程序员经历。全网粉丝12w、csdn博客专家、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战*✌ 🍅文末获取源码联系🍅 👇🏻 精彩专栏推荐订阅👇…...
【Unity】鱼群效果模拟
鱼群效果模拟 文章目录 鱼群效果模拟Boid算法实现方式version1_CPUversion2_GPUversion3_Multilaterationversion4_Bitonic_Sorting (GPU友好)version5_Skinning (TODO) 细节项优化项参考链接 Boid算法 Boid算法是一种模拟群体行…...
Unity使用IL2CPP打包时,我们应该注意什么?如何避免(可以举例说明)
这一篇部分内容在Unity之中如何处理C#底层代码那篇博客之中有提及,接下来对这部分进行补充说明。(请先阅读--Unity之中如何处理C#底层代码) 目录 1 注意点 2 如何避免 1 注意点 IL2CPP 在编译过程中会将 IL(中间语言…...
