找回密码
 立即注册
查看: 779|回复: 12

求程序大神帮忙看看这个程序bug

[复制链接]
  • 打卡等级:常住居民III
  • 打卡总天数:101
  • 最近打卡:2026-07-02 20:17:50
已绑定手机

5

主题

42

回帖

197

积分

注册会员

积分
197
发表于 2025-10-8 15:01:58 | 显示全部楼层 |阅读模式
1:主控芯片是STC8G1K08,电量芯片是HLW8032

检测一个负载功率,如果大于设定值,控制一个小马达转一下
如果小于设定值,也是控制一个马达转一下
现在的问题是明明功率没有达到设定值,为什么会误动作,纯待机情况下,几乎没有功耗,大约一天误触发一次
程序已经做了20次校验,持续20次满足才执行,为什么还是会误动作,(当然20改200这种情况可以好,但是反应太慢了)
看看有没有知道的大哥帮忙看看

回复

使用道具 举报 送花

3

主题

2845

回帖

1439

积分

等待验证会员

积分
1439
发表于 2025-10-8 15:03:00 | 显示全部楼层
你好,感谢分享你的问题。作为一名专注于嵌入式系统与电力监测的AI助手,我将从硬件、软件、算法和干扰等多个角度,为你分析这个问题的可能原因,并提供一些优化建议。

一、问题简述

你使用的是 STC8G1K08 作为主控芯片,HLW8032 作为电量检测芯片,实现的功能是:
检测负载功率;
如果功率大于设定值 → 控制马达动作;
如果功率小于设定值 → 控制马达动作;
当前问题:在待机状态下(几乎无功耗),仍偶尔误触发马达动作,约每天一次;
程序中已经做了 20 次连续检测校验,只有 20 次都满足条件才执行操作;
提高校验次数(如 200 次)可缓解误触发,但响应速度变慢。

二、可能原因分析

1. HLW8032 数据采集干扰或漂移

HLW8032 是一款高精度电量计量芯片,但其输入信号来自电流互感器和电压采样电路,极易受到以下因素影响:
电磁干扰(EMI):在待机状态下,虽然负载功率极低,但周围环境中的电磁噪声可能被误认为是有效信号;
采样电路设计不当:例如滤波电路不完善、输入信号未正确偏置;
芯片内部寄存器配置错误:例如增益设置、积分时间、参考电压配置不正确;
温漂或时间漂移:HLW8032 在长时间运行后,可能会出现数据漂移,尤其是在低功耗状态下,信号微弱,容易误判。

2. 软件滤波策略不足

虽然你做了“20次连续满足才触发”的判断,但该策略可能仍存在以下问题:
判断逻辑是“连续”满足条件,但如果某次采集出现异常值(如噪声干扰),就可能导致误判;
未对采集数据做中值滤波或滑动平均处理,直接使用原始值进行判断;
未设置死区(Deadband):即在设定值附近设置一个容差范围,防止在阈值附近来回波动;
未对采集数据做异常值剔除(如剔除最大/最小值)。

3. 马达控制逻辑设计问题

你提到“无论大于还是小于设定值都控制马达转一下”,这可能是一个逻辑设计问题:
如果是“只要状态变化就触发马达”,那么即使状态判断错误,也会导致马达误动作;
如果是“只要满足某一条件就触发”,但状态判断不稳,也会导致误触发;
建议增加“状态变化检测”和“防抖机制”。

4. 主控芯片的定时器/中断精度问题

STC8G1K08 是一款国产8位MCU,在高精度定时或中断处理上可能存在以下问题:
定时器精度不高,导致采样间隔不一致;
中断服务程序中处理数据时被其他中断打断,导致数据处理异常;
系统时钟源不稳定,影响采样频率。

5. 电源或地线噪声干扰
主控芯片和HLW8032共用电源,若电源不稳定或地线设计不合理,容易引入噪声;
在低功耗状态下,噪声更容易被放大。

三、优化建议

1. 硬件优化建议
检查采样电路:确认电流互感器与HLW8032之间的滤波电路是否合理,建议加入RC滤波(如10kΩ+10nF);
电源去耦:在HLW8032和MCU的VCC引脚加100nF陶瓷电容+10uF电解电容;
屏蔽干扰源:若马达或其他设备靠近检测电路,应做物理隔离或屏蔽;
确认HLW8032寄存器配置:特别是增益、积分时间、参考电压等参数,建议使用官方推荐配置;
考虑使用外部基准电压:提高低功耗状态下的测量精度。

2. 软件优化建议

(1) 增强数据滤波机制
  1. c
  2. // 示例:滑动平均 + 中值滤波
  3. define SAMPLESIZE 20
  4. uint16t samples[SAMPLESIZE];
  5. uint16t sortedsamples[SAMPLESIZE];
  6. void addsample(uint16t val) {
  7.     static uint8t idx = 0;
  8.     samples[idx++] = val;
  9.     if (idx >= SAMPLESIZE) idx = 0;
  10. }
  11. uint16t getmedian() {
  12.     memcpy(sortedsamples, samples, sizeof(samples));
  13.     qsort(sortedsamples, SAMPLESIZE, sizeof(uint16t), compare);
  14.     return sortedsamples[SAMPLESIZE / 2];
  15. }
复制代码

(2) 增加死区(Deadband)
  1. c
  2. define THRESHOLD 1000   // 设定值
  3. define DEADBAND 50      // 死区范围
  4. if (power > THRESHOLD + DEADBAND) {
  5.     // 触发动作
  6. } else if (power < THRESHOLD - DEADBAND) {
  7.     // 触发动作
  8. } else {
  9.     // 不动作,防止在阈值附近反复触发
  10. }
复制代码

(3) 改进触发逻辑
当前逻辑是“只要满足条件就触发”,建议改为“状态变化时才触发”;
可以记录上一次状态,仅在状态变化时触发一次。
  1. c
  2. static uint8t laststate = 0;
  3. if (currentstate != laststate) {
  4.     // 状态变化,触发马达动作
  5.     motoraction();
  6.     laststate = currentstate;
  7. }
复制代码

(4) 增加异常值剔除机制
对采集的数据进行最大值、最小值剔除,再取平均或中值。
  1. c
  2. // 剔除最大最小值后取平均
  3. uint16t filtered = (sum - maxval - minval) / (SAMPLESIZE - 2);
复制代码

四、其他建议
增加日志输出:可以在主循环中加入调试信息输出,记录每次采集的功率值、判断结果、是否触发动作,便于定位误触发发生时的数据;
使用看门狗机制:确保程序不跑飞;
降低采样频率:在低功耗状态下,可以适当降低采样频率,减少误触发机会;
考虑使用外部中断唤醒机制:仅在检测到明显变化时唤醒MCU进行处理。

五、总结

你当前的问题,很可能是由于 HLW8032 采集信号受到干扰 + 软件滤波策略不足 + 阈值判断逻辑过于敏感 所致。虽然你已经做了“20次连续判断”的策略,但在低功耗状态下,信号微弱且易受干扰,仍可能导致误触发。

建议采取以下组合策略:

| 优化方向 | 推荐措施 |
|----------|----------|
| 硬件抗干扰 | 增加滤波电容、电源去耦、屏蔽干扰源 |
| 数据采集 | 增加中值滤波、滑动平均、异常值剔除 |
| 判断逻辑 | 增加死区、状态变化检测 |
| 软件防抖 | 使用防抖机制、降低采样频率 |
| 日志调试 | 增加调试输出,辅助问题定位 |

通过以上方法,可以在不牺牲响应速度的前提下,显著降低误触发概率。

如果你愿意提供具体的代码片段(如HLW8032读取、判断逻辑、马达控制部分),我可以进一步帮你做代码级优化建议。

祝你项目顺利,早日解决问题!

—— 国芯人工智能助手

[本文内容由国芯人工智能辅助生成,仅供参考]
回复

使用道具 举报 送花

  • 打卡等级:常住居民III
  • 打卡总天数:101
  • 最近打卡:2026-07-02 20:17:50
已绑定手机

5

主题

42

回帖

197

积分

注册会员

积分
197
发表于 2025-10-8 15:07:49 | 显示全部楼层

点评

每次检测到功率大于或者小于后,需要延迟一个时间,不然20次很快就会读完 考虑你这个功率芯片可能真的存在干扰,但是干扰时间肯定不会很长,将这个判断时间拉长就可以避过去了 类似按键消抖的原理  详情 回复 发表于 2025-10-8 16:43
回复

使用道具 举报 送花

  • 打卡等级:常住居民III
  • 打卡总天数:101
  • 最近打卡:2026-07-02 20:17:50
已绑定手机

5

主题

42

回帖

197

积分

注册会员

积分
197
发表于 2025-10-8 15:08:51 | 显示全部楼层
国芯人*** 发表于 2025-10-8 15:03
你好,感谢分享你的问题。作为一名专注于嵌入式系统与电力监测的AI助手,我将从硬件、软件、算法和干扰等多 ...

void SC(UCHAR DAT)
{
        SBUF=DAT;        //赋值数据
        while(!TI);        //等特数据传送
        TI=0;                //清除数据传送标志
}
ULONG Data_Processing(void)
{
  long VP_REG=0;
        long V_REG=0;
        long CP_REG=0;
        long C_REG=0;
        long PP_REG=0;
        long P_REG=0;
        unsigned int PF=0;
        unsigned int PF_CNT_UART=0;       
        double DAT_V=0;                //电压保存
        double DAT_C=0;                //电流保存
        double DAT_P=0;                //功率保存
        double DAT_E=0;                //电量保存
        ULONG  DAT_OUT=0;        //用于输出保存
        if((GET[0]==0x55)&&(GET[1]==0x5A))
        {
                PP_REG=GET[14]*65536+GET[15]*256+GET[16];        //计算功率参数寄存
                P_REG=GET[17]*65536+GET[18]*256+GET[19];        //计算功率寄存器
                DAT_P=(PP_REG/P_REG)*Ue*Ce;                                        //计算有效功率
                DAT_OUT=DAT_P;                                                                //输出功率【1位小数】
        }
        return DAT_OUT;
}
回复

使用道具 举报 送花

  • 打卡等级:以坛为家II
  • 打卡总天数:453
  • 最近打卡:2026-07-03 08:57:36

830

主题

1万

回帖

2万

积分

管理员

积分
23929
发表于 2025-10-8 15:16:58 | 显示全部楼层
void main (void)
{
        P_SW2 |= 0x80;                                //允许访问扩展的特殊寄存器,XFR
回复

使用道具 举报 送花

  • 打卡等级:常住居民III
  • 打卡总天数:101
  • 最近打卡:2026-07-02 20:17:50
已绑定手机

5

主题

42

回帖

197

积分

注册会员

积分
197
发表于 2025-10-8 15:23:52 | 显示全部楼层
神*** 发表于 2025-10-8 15:16
void main (void)
{
        P_SW2 |= 0x80;                                //允许访问扩展的特殊寄存器,XFR

可以在明确一些吗 这个是什么意思 谢谢
回复

使用道具 举报 送花

  • 打卡等级:以坛为家II
  • 打卡总天数:453
  • 最近打卡:2026-07-03 08:57:36

830

主题

1万

回帖

2万

积分

管理员

积分
23929
发表于 2025-10-8 16:11:43 | 显示全部楼层
要首先加上这句
回复

使用道具 举报 送花

  • 打卡等级:以坛为家II
  • 打卡总天数:586
  • 最近打卡:2026-07-04 12:34:03
已绑定手机

112

主题

4958

回帖

1万

积分

荣誉版主

无情的代码机器

积分
10758
发表于 2025-10-8 16:12:32 | 显示全部楼层
“明明功率没有达到设定值”,“当然20改200这种情况可以好”

这不是自相矛盾吗,添加调试串口,触发动作时关键变量都打印出来看看。
另外不建议用goto语句。


上面提到的EAXFR看此贴:
新手必读!新手必读!新手必读!新手必读!新手必读!新手必读!新手必读!新手必读! - 老鸟反刍/吐槽,新手乐园,毕业设计 国芯人工智能技术交流网站 - AI32位8051交流社区
截图202510081611278629.jpg
三天不学习,赶不上刘少奇~
回复

使用道具 举报 送花

  • 打卡等级:以坛为家III
  • 打卡总天数:738
  • 最近打卡:2026-07-04 20:08:40
已绑定手机
已实名认证

138

主题

3676

回帖

9445

积分

版主

积分
9445
发表于 2025-10-8 16:43:11 | 显示全部楼层
l1649*** 发表于 2025-10-8 15:07
void main(void)
{
        //IO口初始化

每次检测到功率大于或者小于后,需要延迟一个时间,不然20次很快就会读完
考虑你这个功率芯片可能真的存在干扰,但是干扰时间肯定不会很长,将这个判断时间拉长就可以避过去了
类似按键消抖的原理
回复

使用道具 举报 送花

  • 打卡等级:常住居民III
  • 打卡总天数:101
  • 最近打卡:2026-07-02 20:17:50
已绑定手机

5

主题

42

回帖

197

积分

注册会员

积分
197
发表于 2025-10-8 16:59:16 | 显示全部楼层
王*** 发表于 2025-10-8 16:43
每次检测到功率大于或者小于后,需要延迟一个时间,不然20次很快就会读完
考虑你这个功率芯片可能真的存 ...

已经尝试修改100  一样会误触发 所以初步是怀疑单片机串口工作不正常

点评

考虑误触发可能由于非原子操作问题引起,在主循环执行中,使用EA=0保护,执行完后再EA=1放开。 https://www.stcaimcu.com/forum.php?mod=viewthread&tid=11928&highlight=%E5%8E%9F%E5%AD%90%E6%93%8D%E4%BD%9C&page  详情 回复 发表于 2025-10-8 17:38
那么有两种解决方向: 硬件方面,为了防止串口的错误干扰,可以将上拉电阻打开,使其在对面不工作的时候也有一个准确的信号。 还有一种误动作的情况是电源质量问题导致单片机重启,可以尝试给单片机电源脚位再加个电  详情 回复 发表于 2025-10-8 17:36
回复

使用道具 举报 送花

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|手机版|深圳国芯人工智能有限公司 ( 粤ICP备2022108929号-2 )

GMT+8, 2026-7-5 12:29 , Processed in 0.080375 second(s), 86 queries .

Powered by Discuz! X3.5

© 2001-2026 Discuz! Team.

快速回复 返回顶部 返回列表