STC89C52RC/LE52RC

STC89C52RC

芯片手册

STC89C52

原理图

扩展版原理图

扩展版原理图

功能示例

LED灯

LED灯的常亮效果

c 复制代码
# include <STC89C5xRC.H>
void main(){
		//将LED连接的P00端口设置为0
		P00 =0;
		while(1);
}

LED灯的闪烁

c 复制代码
/*----------------------------------------------------------
 * 文件名:LED_FlowLight.c
 * 功能:STC89C52RC单片机控制的LED流水灯程序(单向左移循环)
 * 硬件连接:P0口接8个共阳LED,P4.6控制蜂鸣器
 * 作者:[您的名字]
 * 日期:[创建日期]
 *---------------------------------------------------------*/

#include <STC89C5xRC.H> // 包含STC89C52RC系列单片机寄存器定义头文件
#include <INTRINS.H>    // 包含内部函数库(提供_nop_()空指令)

/* 类型重定义(增强可读性)*/
typedef unsigned char u8;  // 定义无符号8位数据类型(范围0~255)
typedef unsigned int u16;  // 定义无符号16位数据类型(范围0~65535)

/* 函数声明 */
void Delayms(u16 count);   // 毫秒级延时函数声明

/*----------------------------------------------------------
 * 主函数
 *---------------------------------------------------------*/
void main()
{
    // 变量初始化
    u8 temp = 0x01;  // 初始化LED位置(二进制00000001,对应最右侧LED)
                     // 注:实际是P0.0对应第一个LED,P0.7对应第八个LED
    
    P46 = 0;         // 关闭蜂鸣器(硬件设计缺陷,P4.6低电平关闭蜂鸣器)
                     // 如果不设置,上电时可能产生噪音
    
    while (1)        // 主循环(单片机程序必须包含无限循环)
    {
        /* LED显示控制 */
        P0 = ~temp;  // 输出到P0口控制LED:
                     // - 取反操作是因为采用共阳接法(端口输出0时LED亮)
                     // - 例如temp=0x01(00000001),取反后=0xFE(11111110),
                     //   即P0.0输出0,对应LED点亮
        
        /* 更新LED位置 */
        temp <<= 1;  // 左移一位,实现LED流水效果
                     // 例如:0x01→0x02→0x04→...→0x80
        
        /* 循环检测 */
        if (temp == 0) // 当左移超出8位时(0x80<<1会变成0x00)
        {
            temp = 0x01; // 重新从最右侧开始
        }
        
        /* 延时控制流水速度 */
        Delayms(100); // 延时100ms(控制LED移动速度)
    }
}

/*----------------------------------------------------------
 * 函数名称:Delayms
 * 功能:实现毫秒级延时
 * 参数:count - 需要延时的毫秒数
 * 说明:针对12MHz晶振校准,其他频率需调整参数
 *---------------------------------------------------------*/
void Delayms(u16 count)
{
    /* 变量定义(使用data关键字将变量存储在内部RAM,提高访问速度)*/
    u8 data i, j;  
    
    while (count--)  // 外层循环(控制总延时毫秒数)
    {
        _nop_();    // 空指令(消耗1个机器周期,12MHz下=1us)
                    // 用于微调延时精度
        
        /* 双重循环实现精确延时 */
        i = 2;
        j = 199;
        do
        {
            while (--j); // 内层循环1(约199×3个机器周期)
        } while (--i);   // 内层循环2(外层循环2次)
        
        /* 
         * 延时计算(12MHz时钟):
         * - 1机器周期=1us
         * - 内层循环:199×3 = 597us
         * - 外层循环:2×597 = 1194us ≈ 1ms
         * - 总延时:count × 1ms
         */
    }
}

LED灯的跑马灯效果:从左到右,从右到左

c 复制代码
#include <STC89C5xRC.H>  // 包含STC89C52RC系列单片机头文件
#include <INTRINS.H>     // 包含 intrinsics 函数(如_nop_)

typedef unsigned char u8;   // 定义无符号8位数据类型(0~255)
typedef unsigned int u16;   // 定义无符号16位数据类型(0~65535)

// 函数声明
void Delayms(u16 count);    // 毫秒级延时函数声明

void main()
{
    // 变量初始化
    u8 temp = 0x01;     // 初始灯位(00000001,最右侧LED亮)
    bit is_left = 1;    // 方向标志(1=左移,0=右移)

    while (1)           // 主循环
    {
        P0 = ~temp;     // 输出到P0口(取反因为LED共阳接法)
        
        // 根据移动方向更新灯位
        if (is_left) {
            temp <<= 1; // 左移一位(LED向左移动)
        } else {
            temp >>= 1; // 右移一位(LED向右移动)
        }

        // 检测边界条件并改变方向
        if (temp == 0x80) { // 当移动到最左端(10000000)
            is_left = 0;     // 改为右移方向
        }
        if (temp == 0x01) {  // 当移动到最右端(00000001)
            is_left = 1;      // 改为左移方向
        }

        Delayms(100);    // 延时100ms控制移动速度
    }
}

/**
 * @brief 毫秒级延时函数
 * @param count 延时毫秒数
 * @note 针对12MHz晶振校准,其他频率需要调整参数
 */
void Delayms(u16 count)
{
    u8 data i, j;       // 使用data关键字将变量存储在内部RAM
    while (count--)     // 外层循环(毫秒级)
    {
        _nop_();       // 空指令(4个时钟周期)
        i = 2;
        j = 199;
        do              // 内层循环(微秒级)
        {
            while (--j); // 约100us
        } while (--i);   // 组合成约1ms延时
    }
}

数码管

静态数码管

c 复制代码
#include <STC89C5xRC.H>  // 包含STC89C52RC系列单片机头文件
#include <INTRINS.H>     // 包含 intrinsics 函数(如_nop_)

typedef unsigned char u8;   // 定义无符号8位数据类型
typedef unsigned int u16;   // 定义无符号16位数据类型
typedef unsigned long u32;  // 定义无符号32位数据类型

// 函数声明
void DigitalTube_setBuffer(u32 number);  // 设置数码管显示缓冲区
void DigitalTube_Single(u8 pos, u8 number); // 控制单个数码管显示
void DigitalTube_Refresh();              // 刷新整个数码管显示
static void Delayms(u16 count);         // 毫秒级延时函数(static限制作用域)

// 数码管段选码(共阴数码管0-9,对应a~dp段)
// 编码格式:gfedcba(P0.0~P0.6),最高位P0.7为小数点
const u8 number_codes[10] = {
    0x3F, // 0 - 00111111
    0x06, // 1 - 00000110
    0x5B, // 2 - 01011011
    0x4F, // 3 - 01001111
    0x66, // 4 - 01100110
    0x6D, // 5 - 01101101
    0x7D, // 6 - 01111101
    0x07, // 7 - 00000111
    0x7F, // 8 - 01111111
    0x6F  // 9 - 01101111
};

u8 digital_buffer[8]; // 数码管显示缓冲区(存储8位数码管的段选值)

void main()
{
    // 初始化IO口
    // P0 = 0x00;  // 段选初始化为全灭(注释掉,实际在刷新函数中处理)
    // P1 = 0xC7;  // 位选初始化为全灭(P1.3-P1.5控制位选,11000111)
    P46 = 0;    // 可能的总使能信号(低电平有效)
    P36 = 0;    // 数码管使能信号(低电平有效)
    P34 = 1;    // 关闭流水灯(高电平关闭)
    
    DigitalTube_setBuffer(99998888); // 设置初始显示值为250
    
    while (1)
    {
        DigitalTube_Refresh(); // 持续刷新数码管显示
    }
}

/**
 * @brief 设置数码管显示缓冲区内容
 * @param number 要显示的数字(最大支持8位数)
 * @note 数字将右对齐显示,不显示前导零
 */
void DigitalTube_setBuffer(u32 number)
{
    u8 i;
    
    // 1. 清空缓冲区(全部显示空白)
    for (i = 0; i < 8; i++) {
        digital_buffer[i] = 0; // 0表示不显示任何段
    }
    
    // 2. 从最右侧开始填充数字(右对齐)
    for (i = 7; i >= 0; i--) {
        digital_buffer[i] = number_codes[number % 10]; // 获取当前位的段码
        number /= 10;  // 移除已处理的最低位
        
        if (number == 0) break; // 数字已处理完毕则退出
    }
}

/**
 * @brief 控制单个数码管显示
 * @param pos 数码管位置(0-7对应位选)
 * @param number 要显示的段码值
 * @note 使用P1.3-P1.5控制3-8译码器选择位选
 */
void DigitalTube_Single(u8 pos, u8 number)
{
    // 1. 位选控制(通过P1.3-P1.5)
    pos <<= 3;       // 左移3位,将0-7映射到P1.3-P1.5
    P1 &= 0xC7;      // 11000111 - 清除位选位(P1.3-P1.5)
    P1 |= pos;       // 设置新的位选
    
    // 2. 段选输出
    P0 = number;     // 输出段码值到P0口
}

/**
 * @brief 刷新整个数码管显示(动态扫描)
 * @note 采用循环扫描方式,每位显示1ms
 */
void DigitalTube_Refresh()
{
    u8 i = 0;
    while (i <= 7)  // 扫描0-7共8位数码管
    {
        DigitalTube_Single(i, digital_buffer[i]); // 显示当前位
        Delayms(1);  // 保持显示1ms
        i++;
    }
}

/**
 * @brief 毫秒级延时函数
 * @param count 延时毫秒数
 * @note 使用static限制只在本文件使用,防止命名冲突
 */
static void Delayms(u16 count)
{
    u8 data i, j;
    while (count--)
    {
        _nop_();  // 空指令,用于精确延时
        i = 2;
        j = 199;
        do
        {
            while (--j);  // 内层循环延时
        } while (--i);    // 外层循环延时
    }
}

数码管计数

mian.c

c 复制代码
#include ".\Com\Com_Util.h"
#include ".\Int\Int_DigitalTube.h"

// 主函数
void main()
{
    u8 num = 100;
    u8 count = 0;
    u8 i=0;
    P46 = 0; // 关闭蜂鸣器

    // 初始化数码管
    Int_DigitalTube_Init();
    //设置要显示的数字
    Int_DigitalTube_setBuffer(num);
    //死循环
    while (1)
    {
        Int_DigitalTube_setBuffer(num);
        while (count <= 100)
        {
            Int_DigitalTube_Refresh();
            count++;
        }
        if (num > 0)
        {
            num--;
            count=0;
        }else if (num == 0)
        {
            while (1)
            {
                Int_DigitalTube_setBuffer(0);
            }
        }
    }
}

App.c

App.h

Com.c

c 复制代码
#include "Com_Util.h"

// 延时函数,指定延时多少毫秒
void Delayms(u16 count)
{
    u8 data i, j;
    while (count)
    {
        _nop_();
        i = 2;
        j = 199;
        do
        {
            while (--j)
                ;
        } while (--i);
        count--;
    }
}

Com.h

c 复制代码
#ifndef __COM_UTIL_H__
#define __COM_UTIL_H__

#include <STC89C5xRC.H>
#include <INTRINS.H>

// 类型别名
typedef unsigned char u8;
typedef unsigned int u16;
typedef unsigned long u32;

/**
 * @brief 延时函数,指定延时多少毫秒
 *
 * @param count 指定多少毫秒
 */
void Delayms(u16 count);

#endif

Dir.c

Dir.h

Int.c

c 复制代码
#include "Int_DigitalTube.h"

// 定义数组,保存每个数字的段选信息
static u8 s_number_codes[10] = {
    0x3F, // 0
    0x06, // 1
    0x5B, // 2
    0x4F, // 3
    0x66, // 4
    0x6D, // 5
    0x7D, // 6
    0x07, // 7
    0x7F, // 8
    0x6F  // 9
};

// 定义数组,8个元素,对应数码管8个位置; 每个元素存储数字的段选信息
static u8 s_digital_buffer[8];

/**
 * @brief 数码管初始化
 */
void Int_DigitalTube_Init()
{
    // 打开数码管开关
    P36 = 0;
    // 关闭流水灯
    P34 = 0;
}

/**
 * @brief 将指定的整数设置到数码管显示缓存中(s_digital_buffer数组)
 *
 * @param number
 */
void Int_DigitalTube_setBuffer(u32 number)
{
    u8 i;

    // 1. 清空之前的显示内容
    for (i = 0; i < 8; i++)
    {
        s_digital_buffer[i] = 0x00;
    }

    // 2. 依次取出number中每位上的数,将其段选信息存储数组;最低位存入最后一个元素
    for (i = 7;; i--)
    {
        // 取出当前位上的数,将其段选信息存入数组指定位置
        s_digital_buffer[i] = s_number_codes[number % 10];
        // 处理number,去掉最低位
        number /= 10;
        // 如果number变为0,说明数字已经取完,停止循环
        if (number == 0 || i == 0)
        {
            break;
        }
    }
}

/**
 * @brief 数码管指定位置指定数字
 *
 * @param pos     位置,使用数字0~7分别表示从左边数第1到到第8个
 * @param code    数字的段选信息
 */
void Int_DigitalTube_Single(u8 pos, u8 number_code)
{
    // 1. 位选 -------------------------------
    // 1.1 pos 左移3位,  与P15、P14、P13 对齐
    pos <<= 3;
    // 1.2 将P1的P15、P14、P13三位置0,其他位保持不变, P1 & 0b11000111
    P1 &= 0xC7;
    // 1.3 将pos上的三位有效数, 赋值到 P15、P14、P13 位置上
    P1 |= pos;

    // 2. 段选 --------------------------------
    P0 = number_code;
}

/**
 * @brief 刷新数码管
 *
 */
void Int_DigitalTube_Refresh()
{
    // 循环0到7
    u8 i;
    for (i = 0; i <= 7; i++)
    {
        Int_DigitalTube_Single(i, s_digital_buffer[i]);
        Delayms(1);
    }
}

Int.h

c 复制代码
#ifndef __INT_DIGITALTUBE_H__
#define __INT_DIGITALTUBE_H__

#include "..\Com\Com_Util.h"

/**
 * @brief 数码管初始化
 */
void Int_DigitalTube_Init();

/**
 * @brief 将指定的整数设置到数码管显示缓存中(digital_buffer数组)
 * 
 * @param number 
 */
void Int_DigitalTube_setBuffer(u32 number);

/**
 * @brief 数码管指定位置指定数字
 *
 * @param pos     位置,使用数字0~7分别表示从左边数第1到到第8个
 * @param code    数字的段选信息
 */
void Int_DigitalTube_Single(u8 pos, u8 number_code);

/**
 * @brief 刷新数码管
 *
 */
void Int_DigitalTube_Refresh();

#endif

Mid.c Mid.h

数码管显示小数是

mian.c

c 复制代码
#include ".\Com\Com_Util.h"
#include ".\Int\Int_DigitalTube.h"

void main()
{
    P46=0;
    Int_DigitalTube_Init();

    Int_DigitalTube_setBuffer(65.89);
    while (1)
    {
        Int_DigitalTube_Refresh();
    }
}

App.c

App.h

Com.c

c 复制代码
#include "Com_Util.h"

// 延时函数,指定延时多少毫秒
void Delayms(u16 count)
{
    u8 data i, j;
    while (count)
    {
        _nop_();
        i = 2;
        j = 199;
        do
        {
            while (--j)
                ;
        } while (--i);
        count--;
    }
}

Com.h

c 复制代码
#ifndef __COM_UTIL_H__
#define __COM_UTIL_H__

#include <STC89C5xRC.H>
#include <INTRINS.H>

// 类型别名
typedef unsigned char u8;
typedef unsigned int u16;
typedef unsigned long u32;

/**
 * @brief 延时函数,指定延时多少毫秒
 *
 * @param count 指定多少毫秒
 */
void Delayms(u16 count);

#endif

Dir.c

Dir.h

Int.c

c 复制代码
#include "Int_DigitalTube.h"

// 定义数组,保存每个数字的段选信息
static u8 s_number_codes[10] = {
    0x3F, // 0
    0x06, // 1
    0x5B, // 2
    0x4F, // 3
    0x66, // 4
    0x6D, // 5
    0x7D, // 6
    0x07, // 7
    0x7F, // 8
    0x6F  // 9
};

// 定义数组,8个元素,对应数码管8个位置; 每个元素存储数字的段选信息
static u8 s_digital_buffer[8];

/**
 * @brief 数码管初始化
 */
void Int_DigitalTube_Init()
{
    // 打开数码管开关
    P36 = 0;
    // 关闭流水灯
    P34 = 0;
}

/**
 * @brief 将指定的整数设置到数码管显示缓存中(s_digital_buffer数组)
 *
 * @param number
 */
void Int_DigitalTube_setBuffer(float number)
{
    u8 i;
    u8 number_str_index;
    bit is_dot=0;
    char number_str[17]; 

    // 1. 清空之前的显示内容
    for (i = 0; i < 8; i++)
    {
        s_digital_buffer[i] = 0x00; 
    }
    sprintf(number_str,"%f",number);
    number_str_index = strlen(number_str)-1;
    while (number_str[number_str_index]==48)
    {
        number_str_index--;
    }
    // 2. 依次取出number中每位上的数,将其段选信息存储数组;最低位存入最后一个元素
    for (i = 7;; i--,number_str_index--)
    {
        if(number_str[number_str_index]==46){
            is_dot=1;
            i++;
            continue;
        }
        if (is_dot==1)
        {
            s_digital_buffer[i]=s_number_codes[number_str[number_str_index]-48]+0x80;
            is_dot=0;
        }else{
            s_digital_buffer[i]=s_number_codes[number_str[number_str_index]-48];
        }
        if (i == 0||number_str_index == 0)
        {
            break;
        }
    }
}

/**
 * @brief 数码管指定位置指定数字
 *
 * @param pos     位置,使用数字0~7分别表示从左边数第1到到第8个
 * @param code    数字的段选信息
 */
void Int_DigitalTube_Single(u8 pos, u8 number_code)
{
    // 1. 位选 -------------------------------
    // 1.1 pos 左移3位,  与P15、P14、P13 对齐
    pos <<= 3;
    // 1.2 将P1的P15、P14、P13三位置0,其他位保持不变, P1 & 0b11000111
    P1 &= 0xC7;
    // 1.3 将pos上的三位有效数, 赋值到 P15、P14、P13 位置上
    P1 |= pos;

    // 2. 段选 --------------------------------
    P0 = number_code;
}

/**
 * @brief 刷新数码管
 *
 */
void Int_DigitalTube_Refresh()
{
    // 循环0到7
    u8 i;
    for (i = 0; i <= 7; i++)
    {
        Int_DigitalTube_Single(i, s_digital_buffer[i]);
        Delayms(1);
    }
}

Int.h

c 复制代码
#ifndef __INT_DIGITALTUBE_H__
#define __INT_DIGITALTUBE_H__

#include <STDIO.H>
#include <STRING.H>
#include "..\Com\Com_Util.h"

/**
 * @brief 数码管初始化
 */
void Int_DigitalTube_Init();

/**
 * @brief 将指定的整数设置到数码管显示缓存中(digital_buffer数组)
 * 
 * @param number 
 */
void Int_DigitalTube_setBuffer(float number);

/**
 * @brief 数码管指定位置指定数字
 *
 * @param pos     位置,使用数字0~7分别表示从左边数第1到到第8个
 * @param code    数字的段选信息
 */
void Int_DigitalTube_Single(u8 pos, u8 number_code);

/**
 * @brief 刷新数码管
 *
 */
void Int_DigitalTube_Refresh();

#endif

Mid.c

Mid.h

独立按键

独立按键,控制LED的亮灭

c 复制代码
/**
 * @file 按键控制LED示例
 * @brief 通过SW1按键切换P0.0连接的LED状态
 * @硬件连接:
 *    - P0.0 连接LED(低电平点亮)
 *    - P4.2 连接SW1按键(低电平有效)
 *    - P4.6 控制蜂鸣器(低电平关闭)
 */

 #include ".\Com\Com_Util.h"  // 包含通用工具头文件(含Delayms函数声明)
 // #include ".\Int\Int_DigitalTube.h"  // 已注释掉的数码管驱动头文件
 
 // 按键引脚定义(根据实际硬件连接)
 #define SW1 P42  // 按键1(P4.2)
 #define SW2 P43  // 按键2(P4.3)
 #define SW3 P32  // 按键3(P3.2)
 #define SW4 P33  // 按键4(P3.3)
 
 void main()
 {
     /*---- 初始化部分 ----*/
     P46 = 0;  // 关闭蜂鸣器(硬件设计缺陷,P4.6低电平关闭蜂鸣器)
     P00 = 1;  // 初始化P0.0输出高电平(LED默认熄灭)
 
     /*---- 主循环 ----*/
     while (1)
     {
         // 检测SW1按键按下(低电平触发)
         if(SW1 == 0)
         {
             Delayms(10);  // 延时10ms消抖(消除机械按键抖动)
 
             // 再次检测确认按键按下(防抖验证)
             if(SW1 == 0)
             {
                 // 等待按键释放(避免连续触发)
                 while (SW1 == 0); 
 
                 // 切换P0.0状态(LED取反)
                 P00 = ~P00;  
                 /* 
                  * LED连接说明:
                  * - 当P0.0=0时LED点亮
                  * - 当P0.0=1时LED熄灭
                  * ~取反操作实现状态翻转
                  */
             }
         }
         
         // 可在此处添加其他按键检测逻辑(SW2-SW4)
     }
 }
相关推荐
2301_8003997226 分钟前
STM32 USART串口通信
stm32·单片机·嵌入式硬件
小程同学>o<1 小时前
嵌入式开发之STM32学习笔记day10
经验分享·笔记·stm32·单片机·嵌入式硬件·学习
一枚码农出身的猎头4 小时前
【招聘】硬件工程师/项目经历/产品经理(智能水电表、AIOT硬件方向)
嵌入式硬件·产品经理·智能硬件
岂是尔等觊觎5 小时前
PCB设计教程【入门篇】——电路分析基础-元件数据手册
经验分享·笔记·嵌入式硬件·学习·pcb工艺
MaoXian_n6 小时前
[ARM][汇编] 02.ARM 汇编常用简单指令
arm开发·驱动开发·单片机·嵌入式硬件
厦门辰迈智慧科技有限公司7 小时前
水利数据采集MCU水资源的智能守护者
单片机·嵌入式硬件
真的想上岸啊7 小时前
学习STC51单片机14(芯片为STC89C52RC)
单片机·嵌入式硬件·学习
硬核科技7 小时前
超时处理机制设计:从TICK到回调
单片机·嵌入式硬件·嵌入式·编程·嵌入式软件·软件
无垠的广袤8 小时前
【萤火工场GD32VW553-IOT开发板】ADC电压的LabVIEW采集
c++·单片机·嵌入式硬件·物联网·labview