求程序大神帮忙看看这个程序bug
1:主控芯片是STC8G1K08,电量芯片是HLW8032检测一个负载功率,如果大于设定值,控制一个小马达转一下
如果小于设定值,也是控制一个马达转一下
现在的问题是明明功率没有达到设定值,为什么会误动作,纯待机情况下,几乎没有功耗,大约一天误触发一次
程序已经做了20次校验,持续20次满足才执行,为什么还是会误动作,(当然20改200这种情况可以好,但是反应太慢了)
看看有没有知道的大哥帮忙看看
你好,感谢分享你的问题。作为一名专注于嵌入式系统与电力监测的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读取、判断逻辑、马达控制部分),我可以进一步帮你做代码级优化建议。
祝你项目顺利,早日解决问题!
—— 国芯人工智能助手
[本文内容由国芯人工智能辅助生成,仅供参考]
国芯人工智能 发表于 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;
} void main (void)
{
P_SW2 |= 0x80; //允许访问扩展的特殊寄存器,XFR 神农鼎 发表于 2025-10-8 15:16
void main (void)
{
P_SW2 |= 0x80; //允许访问扩展的特殊寄存器,XFR
可以在明确一些吗 这个是什么意思 谢谢 要首先加上这句
“明明功率没有达到设定值”,“当然20改200这种情况可以好”
这不是自相矛盾吗,添加调试串口,触发动作时关键变量都打印出来看看。
另外不建议用goto语句。
上面提到的EAXFR看此贴:
新手必读!新手必读!新手必读!新手必读!新手必读!新手必读!新手必读!新手必读! - 老鸟反刍/吐槽,新手乐园,毕业设计 国芯人工智能技术交流网站 - AI32位8051交流社区
l164908060 发表于 2025-10-8 15:07
void main(void)
{
//IO口初始化
每次检测到功率大于或者小于后,需要延迟一个时间,不然20次很快就会读完
考虑你这个功率芯片可能真的存在干扰,但是干扰时间肯定不会很长,将这个判断时间拉长就可以避过去了
类似按键消抖的原理
王昱顺 发表于 2025-10-8 16:43
每次检测到功率大于或者小于后,需要延迟一个时间,不然20次很快就会读完
考虑你这个功率芯片可能真的存 ...
已经尝试修改100一样会误触发 所以初步是怀疑单片机串口工作不正常
页:
[1]
2