基于STC12C5A60S2系列1T 8051单片机的数码管显示IIC总线器件AT24C02实现掉电保存计时时间应用

STC12C5A60S2系列1T 8051单片机管脚图

在这里插入图片描述在这里插入图片描述

STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式及配置

在这里插入图片描述

STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式介绍

在这里插入图片描述在这里插入图片描述

液晶显示器LCD1602简单介绍

在这里插入图片描述在这里插入图片描述
DDRAM : 数据显示随机存储器 液晶显示器显示数据是来于数据显示随机存储器
在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述
CGRAM:字符生成随机存储器 用户自定义字符会存储在字符生成存储器
在这里插入图片描述
CGROM:字符库只读存储器 只提供字符 用液晶显示器显示这些字符

总结:不管是CGRAM(字符生成随机存储器)自定义的字符 还是CGROM(字符库只读存储器)原有的字符 写进DDRAM(数据显示随机存储器)内存地址上 才能在液晶显示器屏幕显示出来
在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述

IIC通信简单介绍

在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述

掉电保存数据芯片AT24C02简单介绍

在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述

通过四位共阳数码管显示在掉电保存数据芯片AT24C02内存地址上实现掉电保存计时时间

在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述
main.c

#include "STC12C5A60S2.h"  
#include "AT24C02.h" 
#include "AT24C02OfI2C.h"
#include "Timer0.h"
#include "Key.h"
#include "Digitron.h"
sbit LED = P1^2;//位定义LED灯为单片机P1.2脚
 void main()//主函数
{
   
  Timer0Init();//定时器0的16位定时模式1用12分频定时2ms初始化函数 晶振为12MHz
	DigitronBootDisplay();//数码管开机显示函数
  /****
  //uchar Data;//声明数据变量
  uint BootTime;//声明开机次数变量
  uchar String[10];//声明字符串数组变量
  uchar StringCache[3];//声明字符串缓存数组变量 StringCache[0]存放开始计数标志位变量 StringCache[1]存放第一个计数变量 StringCache[2]存放第二个计数变量
  //Timer0Init();//定时器0的16位定时模式1用12分频定时2ms初始化函数 晶振为12MHz
  LCD1602Init();//液晶显示器初始化函数
  ****/
  /****
  Data = ReadByteFrom24C02(0x50,0x20);//单片机作为主机通过I2C通信从24C02内存地址0x20上读取一个字节 
  LCD1602SetDisplayPosition(0,0);//液晶显示器LCD1602第一行第一个位置显示字符串
  LCD1602WriteData(Data/100 + '0');//液晶显示器LCD1602写数据函数 显示百位数据 为什么数据分解后+'0'? 因为液晶显示器LCD1602要显示数据型字符串 而单片机运算字符串的字符是字符对应的ASCII码 如:字符'0'的ASCII码十进制数是48 字符'1'的ASCII码十进制数是49 字符'2'的ASCII码十进制数是50 .... 字符'9'的ASCII码十进制数是57 而编译器对于'0'会自动视为是'0'的ASCII码 即ASCII码十进制数48 举例:拿Data = 251来分解 则有:Data/100=2 Data/10%10=5 data%10=1 要把数据型251转化成字符串为"251" 可拆开看成'2' '5' '1' 它们对应的ASCII码十进制数分别为50 53 49 而Data/100=2+'0'=2+48=50 Data/10%10=5+'0'=5+48=53 data%10=1+'0'=1+48=49 算出50 53 49这三个ASCII码十进制数分别对上字符'2' '5' '1'的ASCII码十进制数为50 53 49 这就是为什么数据分解后+'0'的原因
  LCD1602WriteData(Data/10%10 + '0');//液晶显示器LCD1602写数据函数 显示十位数据
  LCD1602WriteData(Data%10 + '0');//液晶显示器LCD1602写数据函数 显示个位数据
  NumberTransformedToHexString(Data,String);//无符号字符型数据转化为十六进制数的字符串函数 无符号字符型数据要转化为十六进制数的字符串 可以把无符号字符型数据拆成高四位与低四位来分别转化成表示十六进制数的字符串两位数字符
  LCD1602ShowString(5,0,String);//液晶显示器LCD1602第一行第六个位置显示字符串
  Data++;//数据变量自加
  WriteByteTo24C02(0x50,0x20,Data);//写字节给24C02函数 单片机作为主机通过I2C通信在24C02内存地址0x20上写一个字节 
  LCD1602SetDisplayPosition(0,1);//液晶显示器LCD1602第二行第一个位置显示字符串
  LCD1602WriteData(Data/100 + '0');//液晶显示器LCD1602写数据函数 显示百位数据 为什么数据分解后+'0'? 解释同上
  LCD1602WriteData(Data/10%10 + '0');//液晶显示器LCD1602写数据函数 显示十位数据
  LCD1602WriteData(Data%10 + '0');//液晶显示器LCD1602写数据函数 显示个位数据
  NumberTransformedToHexString(Data,String);//无符号字符型数据转化为十六进制数的字符串函数 无符号字符型数据要转化为十六进制数的字符串 可以把无符号字符型数据拆成高四位与低四位来分别转化成表示十六进制数的字符串两位数字符
  LCD1602ShowString(5,1,String);//液晶显示器LCD1602第二行第六个位置显示字符串
  ****/
  /****
  ReadMultiByteFrom24C02(0x50,0x00,StringCache,10);//单片机作为主机通过I2C通信从24C02内存地址0x00上连续读取十个字节 
  StringCacheTransformedToHexString(StringCache,10,String);//字符串缓存数组中的无符号字符型数据转化为十六进制数的字符串函数
  LCD1602ShowString(1,0,String);//液晶显示器LCD1602第一行第二个位置显示字符串
  for(i = 0;i < sizeof(StringCache);i++)
 {
   StringCache[i] = StringCache[i] + 1;//字符串缓存数组中的数据变量自加
  }
  //WriteMultiByteTo24C02(0x50,0x00,StringCache,10);//单片机作为主机通过I2C通信从24C02内存地址0x00上连续写入10个字节
  PageWriteByteTo24C02(0x50,0x00,StringCache,10);//单片机作为主机通过I2C通信从24C02内存地址0x00上按八个字节为一页连续写入十个字节
  ****/
  /****
  ReadMultiByteFrom24C02(24C02Address,24C02RecordeAddress,StringCache,3)//单片机作为主机通过I2C通信从24C02内存地址0x00上读取三个字节
  if(StringCache[0] != BootCountFlag)//判断是否开过机
 {
   StringCache[0] = BootCountFlag;//没开过机就把开机计数标志位变量包含的数据赋给字符串缓存数组中第一个数据变量
   StringCache[1] = 1;//字符串缓存数组中第二个数据变量缓存第一次开机计数
   StringCache[2] = 0;//字符串缓存数组中第二个数据变量清0
  }
  else//开过机
 {
   StringCache[1]++;//字符串缓存数组中第二个数据变量缓存开机计数
   if(StringCache[1] >= BootMultipleCount)//判断字符串缓存数组中第二个数据变量缓存开机计数变量是否大于开机倍数计数变量
  {
    StringCache[1] = 0;//字符串缓存数组中第二个数据变量清0
    StringCache[2]++;//字符串缓存数组中第三个数据变量在字符串缓存数组中第二个数据变量缓存开机计数超过开机倍数计数变量后进行递进缓存开机计数
    if(StringCache[2] >= BootMultipleCount)//判断字符串缓存数组中第三个数据变量是否大于开机倍数计数变量
   {
     StringCache[1] = 0;//字符串缓存数组中第二个数据变量清0
     StringCache[2] = 0;//字符串缓存数组中第三个数据变量清0
    }
   }
  }
  WriteMultiByteTo24C02(24C02Address,24C02RecordeAddress,StringCache,3)//单片机作为主机通过I2C通信从24C02内存地址0x00上写入三个字节
  BootTime = StringCache[2] * BootMultipleCount + StringCache[1];//开机计数运算公式
  IntegerTransformedToHexString(BootTime,String);//整数转化为十六进制数的字符串函数
  LCD1602ShowString(0,0,String);//液晶显示器LCD1602第一行第一个位置显示字符串 
  while(1);//主循环
  ****/
  /****
  while(1)//主循环
 {
   if(TimerCountFlag == 1)//判断2ms定时计数标志位变量是否置1
  {
    LCD1602WriteCommand(LCD1602DisplayLeftMove);//液晶显示器LCD1602写命令函数 每2ms向左移屏
    TimerCountFlag = 0;//定时计数标志位变量清0
   }
  }
  ****/
  RestoreAT24C02Data(AT24C02Address,AT24C02RecordeAddress);//恢复AT24C02数据函数 相当于读取AT24C02数据
  while(1)//主循环
 {
   
//   DigitronDisplayDataSplit();//数码管显示数据分解函数
	 if(BackupFlag == 1)//判断备份标志位变量是否等于1
  {
   
    BackupFlag = 0;//备份标志位变量清0
//    if(CumulativeTime > 10)//这里作为提示试验 先让累积时间变量取10秒 看情况来设置数值
//   {
   
//     CumulativeTime = 10;//保存累积时间
//    LedTimeCount = 5000;//定义LED时间计数变量
//    }
//    else
//   {
   
    BackupAT24C02Data(AT24C02Address,AT24C02RecordeAddress);//备份AT24C02函数 相当于写入数据给AT24C02
		LED = ~ LED;//LED取反
//     LedTimeCount = 0;//LED时间计数变量清0
//    }
   }
//   if(LedTimeCount != 0)//判断LED时间计数变量是否不等于0
//  {
   
//    LedTimeCount--;//LED时间计数变量自减
//    LED = ~ LED;//LED取反
//   }
  }
 } 

AT24C02.c

#include "AT24C02.h" 
#include "Timer0.h"
#include "AT24C02OfI2C.h" 
#define uchar unsigned char//自定义无符号字符型为uchar
#define uint unsigned int//自定义无符号整数型为uint
uchar StringCache[3];//定义字符串缓存数组变量
 /****
 uchar ReadByteFromAT24C02(uchar IcAddress,uchar Address)//读AT24C02字节函数
{
  uchar Data;//声明数据变量
  I2CStar();//I2C启动函数
  I2CWriteByte(IcAddress << 1);//I2C写字节函数 单片机通过I2C通信写匹配通信地址给AT24C02
  I2CWriteByte(Address);//I2C写字节函数 单片机通过I2C通信写AT24C02的地址
  I2CStar();//I2C启动函数
  I2CWriteByte((IcAddress << 1) | 0x01);//I2C写字节函数 单片机通过I2C通信写入读AT24C02
  Data = I2CReadByte();//I2C读字节函数含有的数据赋给数据变量
  SendAck(1);//I2C发送应答函数 单片机作为主机通过I2C通信只读AT24C02内存地址上的一个字节 
  I2CStop();//I2C停止函数
  return Data;//返回数据变量含有的数据
 }
 ****/ 
  void ReadMultiByteFromAT24C02(uchar IcAddress,uchar Address,uchar *StringCache,uchar StringLength)//读AT24C02多字节函数
{
   
  
  //I2CStar();//I2C启动函数
  //I2CWriteByte(IcAddress << 1);//I2C写字节函数 单片机通过I2C通信写匹配通信地址给AT24C02
   do 
  {
   
    I2CStar();//I2C启动函数
    if(!(I2CWriteByte(IcAddress << 1)))//I2C写字节函数 单片机通过I2C通信写匹配通信地址给AT24C02
   {
   
     break;
    }
    I2CStop();//I2C停止函数
   }while(1);
  I2CWriteByte(Address);//I2C写字节函数 单片机通过I2C通信写AT24C02的地址
  I2CStar();//I2C启动函数
  I2CWriteByte((IcAddress << 1) | 0x01);//I2C写字节函数 单片机通过I2C通信写入读AT24C02
  while(StringLength > 1)//判断字节长度是否大于1
 {
   
   *StringCache++ = I2CReadByte();//I2C读字节函数含有的数据赋给字符串缓存变量
   SendAck(0);//I2C发送应答函数 单片机作为主机通过I2C通信连读AT24C02内存地址上的字节
   StringLength--;//字符串长度自减
  }
  *StringCache = I2CReadByte();//I2C读字节函数含有的数据赋给字符串缓存变量
  SendAck(1);//I2C发送应答函数 单片机作为主机通过I2C通信只读AT24C02内存地址上的一个字节 
  I2CStop();//I2C停止函数
 }
 /****
 void WriteByteToAT24C02(uchar IcAddress,uchar Address,uchar Data)//写字节给AT24C02函数
{
  I2CStar();//I2C启动函数
  I2CWriteByte(IcAddress << 1);//I2C写字节函数 单片机通过I2C通信写匹配通信地址给AT24C02
  I2CWriteByte(Address);//I2C写字节函数 单片机通过I2C通信写AT24C02的地址
  I2CWriteByte(Data);//I2C写字节函数
  I2CStop();//I2C停止函数
 }
 ****/
 /****
 void WriteMultiByteToAT24C02(uchar IcAddress,uchar Address,uchar *StringCache,uchar StringLength)//写多个字节给AT24C02函数
{
  while(StringLength > 0)//判断字节长度是否大于0
 {
   do 
  {
    I2CStar();//I2C启动函数
    if(!(I2CWriteByte(IcAddress << 1)));//I2C写字节函数 单片机通过I2C通信写匹配通信地址给AT24C02
   {
     break;
    }
    I2CStop();//I2C停止函数
   }while(1);
   I2CWriteByte(Address++);//I2C写字节函数 单片机通过I2C通信写AT24C02的地址
   I2CWriteByte(*StringCache++);//I2C写字节函数
   I2CStop();//I2C停止函数
   StringLength--;//字符串长度自减
  }
 }
 ****/
 uchar PageWriteByteToAT24C02(uchar IcAddress,uchar Address,uchar *StringCache,uchar StringLength)//按页写字节给AT24C02函数
{
   
  if((AT24C02MaximumStorageSpace - Address) < StringLength)//判断写入字节数储存量是否大于AT24C02剩余储存空间 
 {
   
   return 0;
  }
  while(StringLength > 0)//判断字节长度是否大于0
 {
   
   do 
  {
   
    I2CStar();//I2C启动函数
    if(!(I2CWriteByte(IcAddress << 1)
  • 15
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值