l164908060 发表于 2025-10-8 15:01:58

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

1:主控芯片是STC8G1K08,电量芯片是HLW8032

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

国芯人工智能 发表于 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) 增强数据滤波机制

c
// 示例:滑动平均 + 中值滤波
define SAMPLESIZE 20
uint16t samples;
uint16t sortedsamples;

void addsample(uint16t val) {
    static uint8t idx = 0;
    samples = val;
    if (idx >= SAMPLESIZE) idx = 0;
}

uint16t getmedian() {
    memcpy(sortedsamples, samples, sizeof(samples));
    qsort(sortedsamples, SAMPLESIZE, sizeof(uint16t), compare);
    return sortedsamples;
}

(2) 增加死区(Deadband)

c
define THRESHOLD 1000   // 设定值
define DEADBAND 50      // 死区范围

if (power > THRESHOLD + DEADBAND) {
    // 触发动作
} else if (power < THRESHOLD - DEADBAND) {
    // 触发动作
} else {
    // 不动作,防止在阈值附近反复触发
}

(3) 改进触发逻辑
当前逻辑是“只要满足条件就触发”,建议改为“状态变化时才触发”;
可以记录上一次状态,仅在状态变化时触发一次。

c
static uint8t laststate = 0;

if (currentstate != laststate) {
    // 状态变化,触发马达动作
    motoraction();
    laststate = currentstate;
}

(4) 增加异常值剔除机制
对采集的数据进行最大值、最小值剔除,再取平均或中值。

c
// 剔除最大最小值后取平均
uint16t filtered = (sum - maxval - minval) / (SAMPLESIZE - 2);

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

五、总结

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

建议采取以下组合策略:

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

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

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

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

—— 国芯人工智能助手

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

l164908060 发表于 2025-10-8 15:07:49


l164908060 发表于 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;                //电量保存
        ULONGDAT_OUT=0;        //用于输出保存
        if((GET==0x55)&&(GET==0x5A))
        {
                PP_REG=GET*65536+GET*256+GET;        //计算功率参数寄存
                P_REG=GET*65536+GET*256+GET;        //计算功率寄存器
                DAT_P=(PP_REG/P_REG)*Ue*Ce;                                        //计算有效功率
                DAT_OUT=DAT_P;                                                                //输出功率【1位小数】
        }
        return DAT_OUT;
}

神农鼎 发表于 2025-10-8 15:16:58

void main (void)
{
        P_SW2 |= 0x80;                                //允许访问扩展的特殊寄存器,XFR

l164908060 发表于 2025-10-8 15:23:52

神农鼎 发表于 2025-10-8 15:16
void main (void)
{
        P_SW2 |= 0x80;                                //允许访问扩展的特殊寄存器,XFR

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

神农鼎 发表于 2025-10-8 16:11:43

要首先加上这句

ercircle 发表于 2025-10-8 16:12:32

“明明功率没有达到设定值”,“当然20改200这种情况可以好”

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


上面提到的EAXFR看此贴:
新手必读!新手必读!新手必读!新手必读!新手必读!新手必读!新手必读!新手必读! - 老鸟反刍/吐槽,新手乐园,毕业设计 国芯人工智能技术交流网站 - AI32位8051交流社区

王昱顺 发表于 2025-10-8 16:43:11

l164908060 发表于 2025-10-8 15:07
void main(void)
{
        //IO口初始化


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

l164908060 发表于 2025-10-8 16:59:16

王昱顺 发表于 2025-10-8 16:43
每次检测到功率大于或者小于后,需要延迟一个时间,不然20次很快就会读完
考虑你这个功率芯片可能真的存 ...

已经尝试修改100一样会误触发 所以初步是怀疑单片机串口工作不正常
页: [1] 2
查看完整版本: 求程序大神帮忙看看这个程序bug