STC8H单片机delay_ms函数闪烁不准?原因是参数溢出!

STC8H单片机delay_ms函数闪烁不准?原因是参数溢出!

问题背景

在基于STC8H单片机编写LED闪烁程序时,期望实现1秒(1000ms)的亮灭间隔,但实际闪烁速度远快于1秒。核心代码如下:

c 复制代码
while(1)
{
    P53 = 1;
    delay_ms(1000);  // 期望延时1秒,实际速度异常
    P53 = 0;
    delay_ms(1000);
}
问题根因

排查发现delay_ms函数的形参类型为unsigned char(U8/uint8_t),而U8类型的取值范围是0~255

  • 当传入1000时,数值超出U8的最大值255,发生数值溢出
  • 溢出后实际参与运算的数值为 1000 % 256 = 232(即仅延时232ms),导致闪烁间隔远小于1秒,视觉上闪烁变快。
解决方案
方案1:拆分延时(快速适配)

不修改原delay_ms函数,将1000ms拆分为4次250ms延时(250≤255,无溢出):

c 复制代码
#include <STC8H.H>
#include "gpio.h"
#include "delay.h"

#define mode_0

void gpio_config()
{
    GPIO_InitTypeDef gpio_init;
#ifdef mode_0
    gpio_init.Mode = GPIO_PullUp;
#endif
    gpio_init.Pin  = GPIO_Pin_3;
    GPIO_Inilize(GPIO_P5, &gpio_init);
}

// 封装1秒延时函数(避免重复写4次)
void delay_1s()
{
    delay_ms(250);
    delay_ms(250);
    delay_ms(250);
    delay_ms(250);
}

void main()
{
    gpio_config();
    P53 = 0;  // 灯灭
    
    while(1)
    {
        P53 = 1;  // 灯亮
        delay_1s();  // 精准延时1秒
        P53 = 0;  // 灯灭
        delay_1s();  // 精准延时1秒
    }
}
方案2:修改delay_ms形参类型(根治)

若想直接传入1000,需将delay_ms的形参改为unsigned int(U16/uint16_t,取值范围0~65535),适配STC8H的delay函数示例:

c 复制代码
// delay.h 头文件修改
#ifndef __DELAY_H
#define __DELAY_H

#include <STC8H.H>
#define FOSC 12  // 匹配单片机实际主频(如12MHz)

// 形参改为unsigned int(U16)
void delay_ms(unsigned int ms);  
void delay_us(unsigned char us);

#endif

// delay.c 实现修改
#include "delay.h"

void delay_us(unsigned char us)
{
    unsigned char i;
    while(us--)
    {
        for(i=0; i<FOSC/3; i++);  // STC8H单周期指令校准
    }
}

// 支持0~65535ms延时,无溢出
void delay_ms(unsigned int ms)
{
    unsigned int i;
    for(i=0; i<ms; i++)
    {
        delay_us(1000);
    }
}

修改后即可直接使用delay_ms(1000),无需拆分。

关键注意事项
  1. 数据类型取值范围

    • U8(unsigned char):0~255
    • U16(unsigned int):0~65535
    • U32(unsigned long):0~4294967295
      延时参数需匹配函数形参的类型范围,避免溢出。
  2. 主频校准
    delay函数内的FOSC需与单片机实际主频(如11.0592MHz、24MHz)一致,否则延时仍会有误差。

  3. 代码健壮性
    建议在封装延时函数时,增加参数范围检查(可选):

    c 复制代码
    void delay_ms(unsigned int ms)
    {
        // 防止传入超大值导致延时异常
        if(ms > 60000) ms = 60000; 
        unsigned int i;
        for(i=0; i<ms; i++)
        {
            delay_us(1000);
        }
    }
总结
  1. 核心问题:delay_ms形参为U8类型时,传入1000会溢出,实际仅延时232ms,导致LED闪烁变快;
  2. 快速解决 :将1000ms拆分为4次250ms延时(delay_ms(250)×4);
  3. 根治方案:将delay_ms的形参改为U16(unsigned int),支持直接传入1000;
  4. 避坑要点:编写延时函数时,务必匹配形参数据类型的取值范围,同时校准单片机主频。
相关推荐
Navigator_Z20 分钟前
LeetCode //C - 1033. Moving Stones Until Consecutive
c语言·算法·leetcode
xiangw@GZ2 小时前
智能锁TouchKey的抗干扰设计【2】-软件算法
嵌入式硬件
iCxhust2 小时前
微机原理实践教程(C语言篇)---A001闪烁灯
c语言·开发语言·汇编·单片机·嵌入式硬件·51单片机·微机原理
一起搞IT吧2 小时前
相机Camera日志实例分析之二十:相机Camx【照片后置4800/5000/6400万拍照】单帧流程日志详解
android·嵌入式硬件·数码相机·智能手机
爱编码的小八嘎2 小时前
C语言完美演绎9-9
c语言
笨笨饿3 小时前
69_如何给自己手搓一个串口
linux·c语言·网络·单片机·嵌入式硬件·算法·个人开发
爱编码的小八嘎4 小时前
C语言完美演绎9-16
c语言
她说彩礼65万5 小时前
C语言 文件
linux·服务器·c语言
FreakStudio7 小时前
MicroPython 内核开发者直接狂喜!这个 Claude 插件市场,把开发全流程做成了「对话式外挂」
python·单片机·嵌入式·面向对象·并行计算·电子diy
天诚智能门锁7 小时前
天诚公租房管控平台CAT.1人脸猫眼智能锁助力青神县公租房管理
人工智能·嵌入式硬件·物联网·智能家居·智能硬件