找回密码
 立即注册
楼主: 乐高

AI8051U教学视频课后小练 | 高质量打卡

[复制链接]
  • 打卡等级:以坛为家I
  • 打卡总天数:270
  • 最近打卡:2026-02-06 08:39:27
已绑定手机

1

主题

59

回帖

602

积分

高级会员

积分
602
发表于 2025-2-28 11:00:45 | 显示全部楼层
香河英*** 发表于 2025-2-28 10:52
不是听课15集后可以申请实验箱吗?

不是听课15集后可以申请实验箱,是发表学习心得 七集 以上,赠送 Ai8051U实验箱。
回复

使用道具 举报 送花

  • 打卡等级:以坛为家I
  • 打卡总天数:270
  • 最近打卡:2026-02-06 08:39:27
已绑定手机

1

主题

59

回帖

602

积分

高级会员

积分
602
发表于 2025-3-1 11:11:55 | 显示全部楼层
第17集  串口简单应用  课后小练

这次的视频没加声音😊


程序只要在冲哥的DEMO  Usart2_RunTask中加一段即可
  1. void Usart2_RunTask(void)
  2. {
  3.         if( Rec_Num >= 6 )                //是否接收到了6位以上的数据
  4.         {
  5.                 if(( Rec_Dat[Rec_Num-1] == '\n' ) && ( Rec_Dat[Rec_Num-2] == '\r' ) )        //末尾判断
  6.                 {
  7.                         if( ( Rec_Dat[Rec_Num-6] == 'O' ) && ( Rec_Dat[Rec_Num-5] == 'P' ) && ( Rec_Dat[Rec_Num-4] == 'E' ) && ( Rec_Dat[Rec_Num-3] == 'N' )  )
  8.                         {
  9.                                 passward[0] = 16;
  10.                                 passward[1] = 16;
  11.                                 passward[2] = 16;
  12.                                 passward[3] = 16;
  13.                                 Uart2_SendStr( "打开成功!\r\n" );
  14.                         }
  15.                         else if( ( Rec_Dat[Rec_Num-7] == 'C' ) && ( Rec_Dat[Rec_Num-6] == 'L' ) && ( Rec_Dat[Rec_Num-5] == 'O' ) && ( Rec_Dat[Rec_Num-4] == 'S' ) && ( Rec_Dat[Rec_Num-3] == 'E' )  )
  16.                         {
  17.                                 passward[0] = 17;
  18.                                 passward[1] = 17;
  19.                                 passward[2] = 17;
  20.                                 passward[3] = 17;
  21.                                 Uart2_SendStr( "关闭成功!\r\n" );
  22.                         }
  23.                         else if( ( Rec_Dat[Rec_Num-9] == 'D' ) && ( Rec_Dat[Rec_Num-8] == 'A' ) &&( Rec_Dat[Rec_Num-7] == 'T' ) && ( Rec_Dat[Rec_Num-6] == '+' ) && ( Rec_Dat[Rec_Num-5] == '1' ) && ( Rec_Dat[Rec_Num-4] == '2' ) && ( Rec_Dat[Rec_Num-3] == '3' )  )
  24.                         {
  25.                                 passward[0] = 17;
  26.                                 passward[1] = 1;
  27.                                 passward[2] = 2;
  28.                                 passward[3] = 3;
  29.                                 Uart2_SendStr( "123\r\n" );
  30.                         }
  31.                         Rec_Num = 0;
  32.                 }
  33.         }
  34. }
复制代码
如果字符比较长,采用与的方式比较两组字符串看起来比较冗长,可以采用字符串比较函数memcmp()来简化,需要包含"string.h"头文件。

  1. #include "string.h"
  2. char *S1 = "OPEN\r\n";
  3. char *S2 = "CLOSE\r\n";
  4. char *S3 = "DAT+123\r\n";
  5. void Usart2_RunTask(void)
  6. {
  7.         if( Rec_Num >= 6 )                //是否接收到了6位以上的数据
  8.         {
  9.                 if(( Rec_Dat[Rec_Num-1] == '\n' ) && ( Rec_Dat[Rec_Num-2] == '\r' ) )        //末尾判断
  10.                 {
  11.                         if(!memcmp(Rec_Dat,S1,6))
  12.                         {
  13.                                 passward[0] = 16;
  14.                                 passward[1] = 16;
  15.                                 passward[2] = 16;
  16.                                 passward[3] = 16;
  17.                                 Uart2_SendStr( "打开成功!\r\n" );
  18.                         }
  19.                         else if(!memcmp(Rec_Dat,S2,7))
  20.                         {
  21.                                 passward[0] = 17;
  22.                                 passward[1] = 17;
  23.                                 passward[2] = 17;
  24.                                 passward[3] = 17;
  25.                                 Uart2_SendStr( "关闭成功!\r\n" );
  26.                         }
  27.                         else if(!memcmp(Rec_Dat,S3,9))
  28.                         {
  29.                                 passward[0] = 17;
  30.                                 passward[1] = 1;
  31.                                 passward[2] = 2;
  32.                                 passward[3] = 3;
  33.                                 Uart2_SendStr( "123\r\n" );
  34.                         }
  35.                         Rec_Num = 0;
  36.                 }
  37.         }
  38. }
复制代码
demo.zip (208.91 KB, 下载次数: 92)




回复

使用道具 举报 送花

  • 打卡等级:以坛为家I
  • 打卡总天数:270
  • 最近打卡:2026-02-06 08:39:27
已绑定手机

1

主题

59

回帖

602

积分

高级会员

积分
602
发表于 2025-3-2 09:27:10 | 显示全部楼层
对单片机的modbus RTU的详细解释(转载)
Modbus 一个工业上常用的通讯协议、一种通讯约定。Modbus协议包括RTU、ASCII、TCP。其中MODBUS-RTU最常用,比较简单,在单片机上很容易实现。虽然RTU比较简单,但是看协议资料、手册说得太专业了,起初很多内容都很难理解。
    所谓的协议是什么?就是互相之间的约定嘛,如果不让别人知道那就是暗号。现在就来定义一个新的最简单协议。例如,

协议: “A” --“LED灭”
           “B” --“报警”
           “C” --“LED亮”
单片机接收到“A”控制一个LED灭,单片机接收到“B”控制报警,单片机接收到“C”控制一个LED亮。那么当收到对应的信息就执行相应的动作,这就是协议,很简单吧。

  先来简单分析一条MODBUS-RTU报文,例如:01  06  00 01  00 17  98 04
    01                06                 00 01                00 17          98 04
  从机地址        功能号          数据地址          数据            CRC校验

这一串数据的意思是:把数据 0x0017(十进制23) 写入 1号从机地址 0x0001数据地址。

先弄明白下面的东西。

1、报文
   一个报文就是一帧数据,一个数据帧就一个报文: 指的是一串完整的指令数据,就像上面的一串数据。

2、CRC校验
意义:例如上面的  98 04  是它前面的数据(01 06 00 01 00 17)通过一算法(见附录2,很简单的)计算出来的结果,其实就像是计算累加和那样。(累加和:就是010600010017加起来的值,然后它的算法就是加法)。
作用:在数据传输过程中可能数据会发生错误,CRC检验检测接收的数据是否正确。比如主机发出01 06 00 01 00 17 98 04,那么从机接收到后要根据01 06 00 01 00 17 再计算CRC校验值,从机判断自己计算出来的CRC校验是否与接收的CRC校验(98 04主机计算的)相等,如果不相等那么说明数据传输有错误这些数据不能要。

3、功能号
  意义:modbus 定义。见附录1。
  作用:指示具体的操作。

MODBUS-RTU
一、一个报文分析
先声明下我们的目的,我们是要两个设备通讯,用的是MODBUS协议。上面简单介绍了:“报文”“CRC校验”“功能号”。

在单片机中拿出一部分内存(RAM)进行两个设备通讯,例如:
截图202503020925353607.jpg
数组后面的注释,说明
OX[20]   代表是输出线圈,用功能码 0x01,0x05,0x0F 访问, 开头地址是 0 (这个后续说明)
IX[20]    代表是输入线圈,用功能码 0x02 访问,             开头地址是 1 (这个后续说明)
另外两个一样的道理。
注意:所谓的“线圈”“寄存器”就是“位变量”“16位变量”,不要被迷惑。之所以称“线圈”我觉得应该是对于应用的设备,MODBUS协议是专门针对485总线设备(例PLC)开发的。

1、主机对从机写数据操作
如果单片机接收到一个报文那么就对报文进行解析执行相应的处理,如上面报文:
    01             06            00 01           00 17          98 04
  从机地址        功能号          数据地址          数据         CRC校验

假如本机地址是 1 ,那么单片机接收到这串数据根据数据计算CRC校验判断数据是否正确,如果判断数据无误,则结果是:
            HoldDataReg[1]  =  0x0017;
MODBUS主机就完成了一次对从机数据的写操作,实现了通讯。

2、主机对从机读数据操作
主机进行读HoldDataReg[1] 操作,则报文是:
    01               03                 00 01           00 01                    D5 CA
从机地址        功能号          数据地址      读取数据个数       CRC校验
那么单片机接收到这串数据根据数据计算CRC校验判断数据是否正确,如果判断数据无误,则结果是:返回信息给主机,返回的信息也是有格式的:
返回内容:  
    01            03            02                     0017                 F8 4A
  从机地址   功能号     数据字节个数    两个字节数据    CRC校验
MODBUS主机就完成了一次对从机数据的读操作,实现了通讯。

回复

使用道具 举报 送花

  • 打卡等级:以坛为家I
  • 打卡总天数:270
  • 最近打卡:2026-02-06 08:39:27
已绑定手机

1

主题

59

回帖

602

积分

高级会员

积分
602
发表于 2025-3-2 09:41:27 | 显示全部楼层
二、MODBUS报文模型
以上了解到了MODBUS的一帧报文是如何通讯的,其实每个报文的格式都基本一样的。
截图202503020942325718.jpg
这里两个缩略词以前不知道,但是现在要明白指的是什么,“ADU”“PDU”
ADU: 应用数据单元
PDU: 协议数据单元

三、MODBUS数据模型
截图202503020942534855.jpg
截图202503020943106545.jpg
四、MODBUS事务处理
  下列状态图描述了在服务器侧MODBUS事务处理的一般处理过程。
截图202503020943283147.jpg
五、MODBUS请求与响应
  看MODBUS协议手册,中文第 10 页开始,英文第 24 页开始。
手册非常详细举例说明了MODBUS协议各个功能号的请求与响应。




回复

使用道具 举报 送花

  • 打卡等级:以坛为家I
  • 打卡总天数:270
  • 最近打卡:2026-02-06 08:39:27
已绑定手机

1

主题

59

回帖

602

积分

高级会员

积分
602
发表于 2025-3-2 09:50:27 | 显示全部楼层
               modbus协议在单片机上实现过程

MODBUS 任务处理函数
截图202503020944563624.jpg
函数中,RcvBuf 为串口接收缓冲区,如果接收了一个报文则,RcvBuf[0] 为从机地址,RcvBuf[1] 为MODBUS功能号。根据功能号做出响应,而具体的操作根据功能号在各自的函数中执行,相当于解析接收到的数据。

附录1:MODBUS-RTU功能码
最常用功能码:
下面“线圈”“寄存器”其实分别直的就是“位变量”“16位变量”
        01 (0x01)        读线圈
        02 (0x02)        读离散量输入
        03 (0x03)        读保持寄存器
        04(0x04)         读输入寄存器
        05 (0x05)        写单个线圈
        06 (0x06)        写单个寄存器
        15 (0x0F)        写多个线圈
        16 (0x10)        写多个寄存器

截图202503020949089010.jpg
附录2:CRC Generation
截图202503020949421525.jpg
转载于:https://www.cnblogs.com/skl374199080/p/3411830.html
回复

使用道具 举报 送花

  • 打卡等级:以坛为家I
  • 打卡总天数:270
  • 最近打卡:2026-02-06 08:39:27
已绑定手机

1

主题

59

回帖

602

积分

高级会员

积分
602
发表于 2025-3-2 09:55:44 | 显示全部楼层
第18集  串口高级应用  课后小练
本人愚钝,MODBUS-RTU 花了整整一个星期的时间才初步完成了此集课后小练,而且还是照搬了梁工的示例程序。
8H/8G/8C/8A系列4组串口同时通信的程序,Modbus 演示程序;可共享T2做波特率发生器 - 串行口,DMA支持的4组串口,RS232,RS485,Modbus, CRC16 国芯技术交流网站 - AI32位8051交流社区
本程序使用USB不间断下载接口,频率选择24M。实验箱J18跳线帽切换到C, D。串口助手更多设置选择“CRC-16 MODBUS协议,发送的数据包结尾增加校验数据”,串口发送接收都选择HEX模式。
J18串口切换跳帽.png 自动添加校验数据.png
  1. u8        MODBUS_RTU(void)
  2. {
  3.         u8        i,j,k;
  4.         u16        reg_addr;                                                                //寄存器地址
  5.         u8        reg_len;                                                                //写入寄存器个数
  6.         u16        crc;
  7.         if(RX1_Buffer[1] == 0x10)                                        //写多寄存器
  8.         {
  9.                 if(RX1_cnt < 9)                return 0x91;                //命令长度错误
  10.                 if((RX1_Buffer[4] != 0) || ((RX1_Buffer[5] *2) != RX1_Buffer[6]))        return 0x92;        //写入寄存器个数与字节数错误
  11.                 if((RX1_Buffer[5]==0) || (RX1_Buffer[5] > REG_LENGTH))        return 0x92;        //写入寄存器个数错误
  12.                 reg_addr = ((u16)RX1_Buffer[2] << 8) + RX1_Buffer[3];        //寄存器地址
  13.                 reg_len = RX1_Buffer[5];                                                                //写入寄存器个数
  14.                 if((reg_addr+(u16)RX1_Buffer[5]) > (REG_ADDRESS1+REG_LENGTH))        return 0x93;        //寄存器地址错误
  15.                 if(reg_addr < REG_ADDRESS1)                return 0x93;                        //寄存器地址错误
  16.                 if((reg_len*2+7) != RX1_cnt)        return 0x91;                        //命令长度错误
  17.                 j = reg_addr - REG_ADDRESS1;        //寄存器数据下标
  18.                 for(k=7, i=0; i<reg_len; i++,j++)
  19.                 {
  20.                         modbus_reg[j] = ((u16)RX1_Buffer[k] << 8) + RX1_Buffer[k+1];        //写入数据, 大端模式
  21.                         k += 2;
  22.                 }
  23.                 if(RX1_Buffer[0] != 0)        //非广播地址则应答
  24.                 {
  25.                         for(i=0; i<6; i++)        TX1_Buffer[i] = RX1_Buffer[i];        //要返回的应答
  26.                         crc = MODBUS_CRC16(TX1_Buffer, 6);
  27.                         TX1_Buffer[6] = (u8)crc;        //CRC是小端模式, 先发低字节,后发高字节。
  28.                         TX1_Buffer[7] = (u8)(crc>>8);
  29.                         B_TX1_Busy = 1;                //标志发送忙
  30.                         TX1_cnt    = 0;                //发送字节计数
  31.                         TX1_number = 8;                //要发送的字节数
  32.                         TI = 1;                                //启动发送
  33.                         
  34.                 }
  35.         }
  36.          else if(RX1_Buffer[1] == 0x06)        //写单寄存器
  37.         {
  38.                 if(RX1_cnt < 6)                return 0x91;                //命令长度错误
  39.                 //if((RX1_Buffer[4] != 0) || ((RX1_Buffer[5] *2) != RX1_Buffer[6]))        return 0x92;        //写入寄存器个数与字节数错误
  40.                 //if((RX1_Buffer[5]==0) || (RX1_Buffer[5] > REG_LENGTH))        return 0x92;        //写入寄存器个数错误
  41.                 reg_addr = ((u16)RX1_Buffer[2] << 8) + RX1_Buffer[3];        //寄存器地址
  42.                 //reg_len = RX1_Buffer[5];        //写入寄存器个数
  43.                 //if((reg_addr+(u16)RX1_Buffer[5]) > (REG_ADDRESS+REG_LENGTH))        return 0x93;        //寄存器地址错误
  44.                 //if(reg_addr != REG_ADDRESS)                return 0x93;        //寄存器地址错误
  45.                 if( RX1_cnt !=6)        return 0x91;        //命令长度错误
  46.                 j = reg_addr - REG_ADDRESS;        //寄存器数据下标
  47.                 modbus_reg[j] = ((u16)RX1_Buffer[4]<<8)+ RX1_Buffer[5];        //写入数据, 大端模式        
  48.                 num = modbus_reg[j];
  49.                 if(RX1_Buffer[0] != 0)        //非广播地址则应答
  50.                 {
  51.                         for(i=0; i<6; i++)        TX1_Buffer[i] = RX1_Buffer[i];        //要返回的应答
  52.                         crc = MODBUS_CRC16(TX1_Buffer, 6);
  53.                         TX1_Buffer[6] = (u8)crc;        //CRC是小端模式, 先发低字节,后发高字节。
  54.                         TX1_Buffer[7] = (u8)(crc>>8);
  55.                         B_TX1_Busy = 1;                //标志发送忙
  56.                         TX1_cnt    = 0;                //发送字节计数
  57.                         TX1_number = 8;                //要发送的字节数
  58.                         TI = 1;                                //启动发送
  59.                 }
  60.         }
  61.         
  62.         else if(RX1_Buffer[1] == 0x03)        //读多寄存器
  63.         {
  64.                 if(RX1_Buffer[0] != 0)        //非广播地址则应答
  65.                 {
  66.                         if(RX1_cnt != 6)                return 0x91;                //命令长度错误
  67.                         if(RX1_Buffer[4] != 0)        return 0x92;        //读出寄存器个数错误
  68.                         if((RX1_Buffer[5]==0) || (RX1_Buffer[5] > REG_LENGTH))        return 0x92;        //读出寄存器个数错误
  69.                         reg_addr = ((u16)RX1_Buffer[2] << 8) + RX1_Buffer[3];        //寄存器地址
  70.                         reg_len = RX1_Buffer[5];        //读出寄存器个数
  71.                         if((reg_addr+(u16)RX1_Buffer[5]) > (REG_ADDRESS1+REG_LENGTH))        return 0x93;        //寄存器地址错误
  72.                         //if(reg_addr < REG_ADDRESS1)                return 0x93;        //寄存器地址错误
  73.                         j = reg_addr - REG_ADDRESS1;        //寄存器数据下标
  74.                         TX1_Buffer[0] = SL_ADDR;        //站号地址
  75.                         TX1_Buffer[1] = 0x03;                //读功能码
  76.                         TX1_Buffer[2] = reg_len*2;        //返回字节数
  77.                         for(k=3, i=0; i<reg_len; i++,j++)
  78.                         {
  79.                                 TX1_Buffer[k++] = (u8)(modbus_reg[j] >> 8);        //数据为大端模式
  80.                                 TX1_Buffer[k++] = (u8)modbus_reg[j];
  81.                         }
  82.                         crc = MODBUS_CRC16(TX1_Buffer, k);
  83.                         TX1_Buffer[k++] = (u8)crc;        //CRC是小端模式, 先发低字节,后发高字节。
  84.                         TX1_Buffer[k++] = (u8)(crc>>8);
  85.                         B_TX1_Busy = 1;                //标志发送忙
  86.                         TX1_cnt    = 0;                //发送字节计数
  87.                         TX1_number = k;                //要发送的字节数
  88.                         TI = 1;                                //启动发送
  89.                 }
  90.         }
  91.         else        return 0x90;        //功能码错误
  92.         return 0;        //解析正确
  93. }
复制代码
示例代码.zip (131.49 KB, 下载次数: 117)   示例代码把数码管数据前的无效零位消隐了。
     









回复

使用道具 举报 送花

  • 打卡等级:以坛为家I
  • 打卡总天数:270
  • 最近打卡:2026-02-06 08:39:27
已绑定手机

1

主题

59

回帖

602

积分

高级会员

积分
602
发表于 2025-3-8 14:22:48 | 显示全部楼层
第十九集  ADC     课间思考

按键显示.png
冲哥的意思是不是要求按下按键松开后,数码管显示键值保持不变,直到下一次按键按下?我的理解是这样的。
程序中增加KEY_Task(void)函数。
  1. void KEY_Task(void)
  2. {
  3.         u8 temp;       
  4.         temp =ADC_KEY_READ(ADC_Read(0));
  5.         if(temp)
  6.         {
  7.                 Key_Vol++;
  8.                 if( Key_Vol==5 )
  9.                 {
  10.                         key = temp;
  11.                 }
  12.         }
  13.                 else
  14.         {
  15.                 Key_Vol = 0;
  16.         }
  17. }       
复制代码
demo1.zip (229.74 KB, 下载次数: 108)

回复

使用道具 举报 送花

  • 打卡等级:以坛为家I
  • 打卡总天数:270
  • 最近打卡:2026-02-06 08:39:27
已绑定手机

1

主题

59

回帖

602

积分

高级会员

积分
602
发表于 2025-3-9 12:04:43 | 显示全部楼层
第十九集  ADC  课后小练

19集 简易电压表.png IMG_20250309_120340.jpg
定义P10引脚电压就是ADC按键端的电压为V_key,按键端ADC读取数值为value,MCU的电源电压为VCC(来自ME6231C33M5G稳压管,输出电压3.3V),MCU内部参考信号为BGV
反推ADC输入电压计算公式1.png 内部1.19V参考信号源.png MCU电源图.png
根据反推工作电压计算公式得到:VCC=4096*BGV/ADC_Read(15);VCC/4096=V_key/value;V_key=VCC*value/4096。
  1. u16 BGV_Read(void)
  2. {
  3.         u16 bgv = 0;
  4.         
  5.         
  6.         bgv = (((u16)CHIPID7) << 8) + (CHIPID8);        //获取内部参考信号源值
  7.         
  8.         return bgv;
  9. }
复制代码
  1. void KEY_Task(void)
  2. {
  3.         u8 temp;        
  4.         temp =ADC_KEY_READ(ADC_Read(0));
  5.         if(temp)
  6.         {
  7.                 Key_Vol++;
  8.                 if( Key_Vol==5 )
  9.                 {
  10.                         key = temp;
  11.                         value = ADC_Read(0);
  12.                 }
  13.         }
  14.                 else
  15.         {
  16.                 Key_Vol = 0;
  17.         }
  18.           VCC = 4096l*BGV_Read()/ADC_Read(15);
  19.           V_key = (u32)VCC*value/4096;
  20.           SEG_Show_U32( key,V_key);
  21.                 //SEG_Show_U32( key,VCC);
  22.                 //SEG_Show_U32( key,value);        
  23.                 //SEG_Show_U32( key,ADC_Read(15));        
  24.                 //SEG_Show_U32( key,BGV_Read());
  25. }        
复制代码

demo.zip (225.92 KB, 下载次数: 118)
回复

使用道具 举报 送花

  • 打卡等级:以坛为家I
  • 打卡总天数:270
  • 最近打卡:2026-02-06 08:39:27
已绑定手机

1

主题

59

回帖

602

积分

高级会员

积分
602
发表于 2025-3-9 20:25:36 | 显示全部楼层
第十九集  ADC  课后小练   程序修正

在上一个程序中,数码管显示的P10端口电压一直在跳动,排查发现是ADC_Read(15)的数值一直在变化,如果电源电压稳定,这应该是一个相对固定的值。
我考虑在程序开始的时候就把它读出来,采用连续读取20次求取平均值。修改后的结果数码管显示稳定。程序中4096后面是小写的“L”,不是数字“1”,用来转换后面的U16为U32长整型。
  1. /*  鉴于前所写程序中ADC_Read(15)的数值一直在跳动,而这个数值应该是一个相对固定值,
  2.         所以在程序运行的开始就把它读取出来,连续读取20次,求其平均值*/        
  3.         Delay10ms();
  4.         Delay10ms();
  5.         for( i=0;i<20;i++)                                                
  6.         {
  7.            BGV_ADC_Read += ADC_Read(15);
  8.            Delay10ms();
  9.         }
  10.         BGV_ADC_Read =BGV_ADC_Read/20;
  11.         VCC = 4096l*BGV_Read()/BGV_ADC_Read;
复制代码

demo2.zip (222.2 KB, 下载次数: 95)
回复

使用道具 举报 送花

  • 打卡等级:以坛为家I
  • 打卡总天数:270
  • 最近打卡:2026-02-06 08:39:27
已绑定手机

1

主题

59

回帖

602

积分

高级会员

积分
602
发表于 2025-3-12 16:04:42 | 显示全部楼层
第20集  ADC_NTC测温  课后小练

屏幕截图 2025-03-12 102857.png IMG_20250312_164149.jpg
NTC温度读取采用冲哥STC32G视频教程第19集NTC温度采集程序。 程序下载运行IRC频率24M,插上测温跳帽JUP2。
  1. void KEY_Task(void)
  2. {
  3.         u8 temp;
  4.         temp = ADC_KEY_READ(ADC_Read(0));        
  5.         if(temp)
  6.         {
  7.                 Key_Vol++;
  8.                 if( Key_Vol==5 )
  9.                 {
  10.                         key = temp;
  11.                         if(key==8)                        //“设置”按键
  12.                                 Set_t = 1;                //按下设置键,标志位Set_t置一
  13.                         if(key==16)                        //“确认退出”按键
  14.                                 Set_t = 0;
  15.                         if(Set_t)
  16.                         {
  17.                                 switch(key)     
  18.                                 {        
  19.                                         //温度增减按键
  20.                                         case 1:   if(baojing[0]<9)  baojing[0]++; break;      
  21.                                         case 2:   if(baojing[1]<9)  baojing[1]++; break;
  22.                                         case 3:   if(baojing[2]<9)  baojing[2]++; break;
  23.                                         case 4:   if(baojing[3]<9)  baojing[3]++; break;
  24.                                         case 9:   if(baojing[0]>0)  baojing[0]--; break;
  25.                                         case 10:  if(baojing[1]>0)  baojing[1]--; break;
  26.                                         case 11:  if(baojing[2]>0)  baojing[2]--; break;
  27.                                         case 12:  if(baojing[3]>0)  baojing[3]--; break;
  28.                                         default: break;
  29.                                 }                        
  30.                         }
  31.                 }
  32.         }
  33.                 else
  34.         {
  35.                 Key_Vol = 0;
  36.         }
  37.         BJ_wendu = baojing[3]*1000+baojing[2]*100+baojing[1]*10+baojing[0];   //报警温度放在数组baojing[]内
  38. }
  39. void NTC_Task(void)    //间隔500ms读取一次NTC值,兼作后四位数码管闪烁计时来源
  40. {
  41.         NTC_wendu = Temp_Cal( ADC_Read(3));    //NTC温度读取采用冲哥STC32G视频教程第19集NTC温度采集程序
  42.         VCC = 4096l*BGV_Read()/BGV_ADC_Read;   //MCU电压计算
  43.         SEG_Show(VCC,NTC_wendu);        //数码管显示,前四位显示电压,后四位显示NTC温度
  44.         if(Set_t)                                //按下设置键,标志位Set_t置一,后四位数码管闪烁
  45.                 blink = ~blink;
  46. }
复制代码
  1. void SEG_Task(void)
  2. {
  3.         if( Seg_no ==0 )                                                               
  4.         {
  5.                 Display_Seg( SEG_NUM[passward[0]]+0x80 , ~T_NUM[0]); //+0x80是为了添加小数点,编译会出现警告C188,可忽略
  6.         }
  7.         else if( Seg_no ==1 )                                                        
  8.         {
  9.                 Display_Seg( SEG_NUM[passward[1]] , ~T_NUM[1]);               
  10.         }        
  11.         else if( Seg_no ==2 )                                                        
  12.         {
  13.                 Display_Seg( SEG_NUM[passward[2]] , ~T_NUM[2]);               
  14.         }        
  15.         else if( Seg_no ==3 )                                                        
  16.         {
  17.                 Display_Seg( SEG_NUM[passward[3]] , ~T_NUM[3]);               
  18.         }
  19.         else if( Seg_no ==4 )
  20.         {
  21.                 if(passward[4]==0)
  22.                         Display_Seg( SEG_NUM[17] , ~T_NUM[0]);      //第一位如果是零,则消隐        
  23.                 else
  24.                         Display_Seg( SEG_NUM[passward[4]], ~T_NUM[4]);               
  25.         }        
  26.         else if( Seg_no ==5 )
  27.         {
  28.                 Display_Seg( SEG_NUM[passward[5]] , ~T_NUM[5]);               
  29.         }        
  30.         else if( Seg_no ==6 )
  31.         {
  32.                 if(Set_t )                                                          //按下设置键,后四位数码管闪烁
  33.                 {
  34.                         if(!blink)
  35.                                 Display_Seg( 0 , ~T_NUM[6]);
  36.                         else
  37.                                 Display_Seg( SEG_NUM[passward[6]]+0x80 , ~T_NUM[6]);
  38.                 }
  39.                 else if(NTC_wendu < BJ_wendu )                                                               
  40.                         Display_Seg( SEG_NUM[passward[6]]+0x80 , ~T_NUM[6]);
  41.                 else
  42.                         Display_Seg( SEG_NUM[passward[6]] , ~T_NUM[6]);
  43.         }
  44.         else if( Seg_no ==7 )
  45.         {
  46.                 Display_Seg( SEG_NUM[passward[7]] , ~T_NUM[7]);               
  47.         }        
  48.         else
  49.         {
  50.                
  51.         }
  52.         Seg_no ++;
  53.         if( Seg_no>7 )
  54.                 Seg_no=0;
  55. }
  56. void SEG_Show(u16 vol,u16 wendu)
  57. {
  58.         u8 i;
  59.         for(i=0;i<4;i++)
  60.         {
  61.                 if(Set_t)                                                       //按下设置键,后四位数码管闪烁
  62.                 {
  63.                         if(blink)
  64.                                 passward[7-i] = 17;
  65.                         else
  66.                                 passward[7-i] = baojing[i];
  67.                         
  68.                 }
  69.                 else
  70.                 {
  71.                         
  72.                         if(wendu > BJ_wendu)                      //如果温度高于预警温度值,则显示“----”
  73.                                 passward[7-i] = 16;
  74.                         else
  75.                         {
  76.                                 passward[7-i] = wendu%10;;
  77.                                 wendu /= 10;
  78.                         }
  79.                 }
  80.         }
  81.         for(i=4;i<8;i++)
  82.         {
  83.                 passward[7-i] = vol%10;
  84.                 vol /= 10;
  85.         }        
  86. }
复制代码


demo.zip (252.02 KB, 下载次数: 135)

回复

使用道具 举报 送花

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

本版积分规则

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

GMT+8, 2026-2-12 21:43 , Processed in 2.131866 second(s), 90 queries .

Powered by Discuz! X3.5

© 2001-2026 Discuz! Team.

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