当前位置: 首页 > ops >正文

结构体指针与非指针 问题及解决

问题描述

第一段位于LCD.h和LCD.c中,

        定义个一个结构体lcd_params,并直接给与指针名*p_lcd_params;

我发现我在调用这个结构体时,即在LCD.c中,使用指针类型定义的

static p_lcd_params p_array_lcd[LCD_NUM];
static p_lcd_params g_p_lcd_selected;.

并且可以正常运行

第二段代码位于Timer.c中

        定义个一个结构体timer_desc,给与指针名ptimer_desc;

我在调用这个结构体时,使用指针类型会有错误,只能使用结构体名定义

timer_desc timer_arry[Timer_NUM];

//LCD.H
typedef struct lcd_params
{
xxxxx
} lcd_params, *p_lcd_params;//LCD.C
static p_lcd_params p_array_lcd[LCD_NUM];
static p_lcd_params g_p_lcd_selected;int register_lcd(p_lcd_params plcd)
{int i;for (i = 0; i < LCD_NUM; i++){if (!p_array_lcd[i]){p_array_lcd[i] = plcd;return i;}}return -1;
}

typedef void (*timer_func)(void);
typedef struct timer_desc
{char *name;timer_func fp;
} timer_desc, *ptimer_desc;timer_desc timer_arry[Timer_NUM];int register_timer(char *name, timer_func fp)
{int i;for (i = 0; i < Timer_NUM; i++){if (!timer_arry[i].fp){timer_arry[i].name = name;timer_arry[i].fp = fp;return 0;}}return -1;
}

问题原因:

在第一段代码中,我在使用p_array_lcd[i]时,在外部调用 register_lcd 的时候传入了结构体的地址:即这个函数他传入了一个结构体地址

int register_lcd(p_lcd_params plcd) 

lcd_params my_lcd = { ... };
register_lcd(&my_lcd);  // 传入结构体地址

而在我的第二段代码中,我并没有为这个结构体指针分配地址。而如果他要是用指针的话

需要使用malloc来为结构体分配内存,或者按照第一种来为他传入结构体地址。

timer_arry[i] = (ptimer_desc)malloc(sizeof(timer_desc));

3:访问方式问题

如果使用指针来定义的话,在访问结构体内部元素时,要使用->来访问

ptimer_desc timer_arry[Timer_NUM];  // 一个结构体指针数组
timer_arry[i]->fp   // 正确 ✅
timer_arry[i]->name // 正确 ✅

不使用结构体来定义的话,使用.来访问结构体元素

timer_desc timer_arry[Timer_NUM];  // 一个结构体指针数组
timer_arry[i].fp   // 正确 ✅
timer_arry[i].name // 正确 ✅

4:第一类代码不使用结构体指针

这个表示:数组中每个元素是一个指向 lcd_params 结构体的指针。你可以在别处创建结构体,然后通过 & 传入:

static p_lcd_params p_array_lcd[LCD_NUM]; // 数组元素是结构体指针

lcd_params lcd1 = { ... }; register_lcd(&lcd1); // ✅ 传入地址,存入指针数组


如果不使用指针,而是改成结构体数组?

static lcd_params lcd_array[LCD_NUM]; // 数组元素是结构体本体 static int lcd_count = 0;

然后 register_lcd 函数可以这样写:

int register_lcd(lcd_params *plcd)

{

if (lcd_count >= LCD_NUM)

return -1;

lcd_array[lcd_count] = *plcd; // 拷贝整个结构体

return lcd_count++;

}


这样做的优点:

  1. 不需要动态分配内存(malloc),更简单、更安全。

  2. 所有结构体数据都存储在本地数组中,便于统一管理。

  3. 避免指针悬空、内存泄漏等问题。

这样做的缺点:

  1. 结构体会被拷贝一次,如果结构体很大(比如包含字符串、数组等),效率会略低。

  2. 如果你希望多个地方共享同一个结构体的数据,就做不到了,因为你拷贝的是副本。

  3. 结构体中如果有 char *void * 成员,还要小心浅拷贝问题(拷贝指针但不是数据本体)。

🆚 指针 VS 非指针对比总结:

项目使用结构体指针不使用结构体指针(结构体数组)
内存管理需自己分配、释放(可能内存泄漏)编译期静态分配,自动管理
数据共享可在多个模块间共享同一结构体是拷贝副本,无法共享
拷贝开销拷贝指针,开销小拷贝整个结构体,开销可能更大
安全性易出错(空指针、悬空指针)更安全,逻辑更直观
灵活性灵活,可实现动态注册、释放等功能结构固定,适合数据量已知、场景简单
http://www.xdnf.cn/news/12088.html

相关文章:

  • To be or Not to be, That‘s a Token——论文阅读笔记——Beyond the 80/20 Rule和R2R
  • centos升级内核
  • LuaJIT2.1 和 Lua5.4.8 性能对比
  • C++网络通信:TCP数据包的发送与接收编程
  • PyCharm中运行.py脚本程序
  • Spring Cloud Gateway 中自定义验证码接口返回 404 的排查与解决
  • 基于BI PaaS架构的衡石HENGSHI SENSE平台技术解析:重塑企业级数据分析基座
  • 猎板厚铜PCB工艺能力如何?
  • Web 架构相关文章目录(持续更新中)
  • Jenkins | Linux环境部署Jenkins与部署java项目
  • select、poll、epoll 与 Reactor 模式
  • 70年使用权的IntelliJ IDEA Ultimate安装教程
  • 代码中文抽取工具并替换工具(以ts为例)
  • 数据结构:泰勒展开式:霍纳法则(Horner‘s Rule)
  • DeepSeek-R1-0528-Qwen3-8B为底座微调领域大模型准备:制作领域专用数据集
  • HarmonyOS:Counter计数器组件
  • QQ邮箱发送验证码(Springboot)
  • AI系统负载均衡与动态路由
  • 力扣HOT100之二分查找: 34. 在排序数组中查找元素的第一个和最后一个位置
  • 数学建模-嘉陵江铊污染事件解题全过程文档及程序
  • 联软NSPM自动化策略管理 助力上交所加速国产化替代提升运维效率
  • matlab实现DBR激光器计算
  • 全维度测试通过!DolphinScheduler 3.2.0单节点部署与验证实录
  • cursor-free-vip使用
  • [实际项目2] 从西门子PLC中读取曲线数值并绘图
  • 半监督学习:低密度分离假设 (Low-Density Separation Assumption)
  • IT组织转型记:从运维支持到数字利润中心的蜕变
  • OpenCV CUDA模块图像处理-----对图像执行 均值漂移过程(Mean Shift Procedure)函数meanShiftProc()
  • opencv学习笔记1:图像基础、图像操作、直方图均衡化详解
  • 破局新能源消纳难题!安科瑞智慧能源平台助力10KV配电网重构未来