基于STC12C5A60S2系列1T 8051单片机的数码管显示IIC总线器件AT24C02记录单片机上电次数应用

基于STC12C5A60S2系列1T 8051单片机的数码管显示IIC总线器件AT24C02记录单片机上电次数应用

  • STC12C5A60S2系列1T 8051单片机管脚图
  • STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式及配置
  • STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式介绍
  • 液晶显示器LCD1602简单介绍
  • IIC通信简单介绍
  • 掉电保存数据芯片AT24C02简单介绍
    • 通过数码管显示掉电保存数据芯片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"
#define uchar unsigned char//自定义无符号字符型为uchar
#define uint unsigned int//自定义无符号整数型为uint
sbit LED = P1^5;//位定义LED灯为单片机P1.2脚
//uchar StringCache[3];//定义字符串缓存数组变量
//uchar PowerOnCountFlag = 0;//定义开机计数标志位变量为0 
uint PowerOnCount = 0;//定义开机计数变量void main()//主函数
{
//	PowerOnCount = PowerOnCountArray[0];// Timer0Init();//定时器0的16位定时模式1用12分频定时2ms初始化函数 晶振为12MHzDigitronBootDisplay();//数码管开机显示函数/****//uchar Data;//声明数据变量uint BootTime;//声明开机次数变量uchar String[10];//声明字符串数组变量uchar StringCache[3];//声明字符串缓存数组变量 StringCache[0]存放开始计数标志位变量 StringCache[1]存放第一个计数变量 StringCache[2]存放第二个计数变量//Timer0Init();//定时器0的16位定时模式1用12分频定时2ms初始化函数 晶振为12MHzLCD1602Init();//液晶显示器初始化函数****//****Data = ReadByteFromAT24C02(0x50,0x20);//单片机作为主机通过I2C通信从AT24C02内存地址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++;//数据变量自加WriteByteToAT24C02(0x50,0x20,Data);//写字节给AT24C02函数 单片机作为主机通过I2C通信在AT24C02内存地址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第二行第六个位置显示字符串****//****ReadMultiByteFromAT24C02(0x50,0x00,StringCache,10);//单片机作为主机通过I2C通信从AT24C02内存地址0x00上连续读取十个字节 StringCacheTransformedToHexString(StringCache,10,String);//字符串缓存数组中的无符号字符型数据转化为十六进制数的字符串函数LCD1602ShowString(1,0,String);//液晶显示器LCD1602第一行第二个位置显示字符串for(i = 0;i < sizeof(StringCache);i++){StringCache[i] = StringCache[i] + 1;//字符串缓存数组中的数据变量自加}//WriteMultiByteToAT24C02(0x50,0x00,StringCache,10);//单片机作为主机通过I2C通信从AT24C02内存地址0x00上连续写入10个字节PageWriteByteToAT24C02(0x50,0x00,StringCache,10);//单片机作为主机通过I2C通信从AT24C02内存地址0x00上按八个字节为一页连续写入十个字节****//****ReadMultiByteFromAT24C02(AT24C02Address,AT24C02RecordeAddress,StringCache,3)//单片机作为主机通过I2C通信从AT24C02内存地址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;//字符串缓存数组中第二个数据变量清0StringCache[2]++;//字符串缓存数组中第三个数据变量在字符串缓存数组中第二个数据变量缓存开机计数超过开机倍数计数变量后进行递进缓存开机计数if(StringCache[2] >= BootMultipleCount)//判断字符串缓存数组中第三个数据变量是否大于开机倍数计数变量{StringCache[1] = 0;//字符串缓存数组中第二个数据变量清0StringCache[2] = 0;//字符串缓存数组中第三个数据变量清0}}}WriteMultiByteToAT24C02(AT24C02Address,AT24C02RecordeAddress,StringCache,3)//单片机作为主机通过I2C通信从AT24C02内存地址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数据
//  ReadByteFromAT24C02(AT24C02Address,AT24C02RecordeAddress);//读AT24C02字节函数PowerOnCount++;//开机计数变量自加while(1)//主循环{
//   if(BackupFlag == 1)//判断备份标志位变量是否等于1
//  {
//    BackupFlag = 0;//备份标志位变量清0
//		PowerOnCount++;
//    if(CumulativeTime > 10)//这里作为提示试验 先让累积时间变量取10秒 看情况来设置数值
//   {
//     CumulativeTime = 10;//保存累积时间
//     LedTimeCount = 5000;//定义LED时间计数变量
//    }
//    else
//   {
//     BackupAT24C02Data(AT24C02Address,AT24C02RecordeAddress);//备份AT24C02函数 相当于写入数据给AT24C02
//     LedTimeCount = 0;//LED时间计数变量清0
//    }
//   }
//   if(LedTimeCount != 0)//判断LED时间计数变量是否不等于0
//  {
//    LedTimeCount--;//LED时间计数变量自减
//    LED = ~ LED;//LED取反
//   }BackupAT24C02Data(AT24C02Address,AT24C02RecordeAddress);//备份AT24C02函数 相当于写入数据给AT24C02
//	 WriteMultiByteToAT24C02(AT24C02Address,AT24C02RecordeAddress,StringCache,3);//写多个字节给AT24C02函数
//   WriteByteToAT24C02(AT24C02Address,AT24C02RecordeAddress,PowerOnCount);//写字节给AT24C02函数 
//}	}	 
//   DigitronDisplayDataSplit();//数码管显示数据分解函数
//  }} 

AT24C02.c

#include "AT24C02.h" 
#include "Timer0.h"
#include "AT24C02OfI2C.h" 
#include "Digitron.h"
#define uchar unsigned char//自定义无符号字符型为uchar
#define uint unsigned int//自定义无符号整数型为uint
uchar StringCache[3];//定义字符串缓存数组变量
extern uint PowerOnCount;//引用外部开机计数变量 
// 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通信写匹配通信地址给AT24C02do {I2CStar();//I2C启动函数if(!(I2CWriteByte(IcAddress << 1)))//I2C写字节函数 单片机通过I2C通信写匹配通信地址给AT24C02{break;}I2CStop();//I2C停止函数}while(1);I2CWriteByte(Address);//I2C写字节函数 单片机通过I2C通信写AT24C02的地址I2CStar();//I2C启动函数I2CWriteByte((IcAddress << 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/591548.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

宏景eHR showmediainfo SQL注入漏洞复现

0x01 产品简介 宏景eHR人力资源管理软件是一款人力资源管理与数字化应用相融合,满足动态化、协同化、流程化、战略化需求的软件。 0x02 漏洞概述 宏景eHR showmediainfo接口处存在SQL注入漏洞,未经过身份认证的远程攻击者可利用此漏洞执行任意SQL指令,从而窃取数据库敏感…

python--不死兔子问题

def rabbit(n):if n < 3:return 1return rabbit(n - 1) rabbit(n - 3)if __name__ __main__:print(rabbit(4))

AWS入门实践-S3 跨区域复制

Amazon S3的跨区域复制&#xff08;Cross-Region Replication, CRR&#xff09;是一项功能&#xff0c;允许你自动、异步地复制存储桶中的对象到一个位于不同AWS区域的存储桶中。这项功能对于数据备份、地理冗余、数据本地化以及合规性要求等场景非常有用。 一、跨区域复制的工…

一致性hash问题(负载均衡原理)

一致性哈希问题 简介 一致性Hash是一种特殊的Hash算法&#xff0c;由于其均衡性、持久性的映射特点&#xff0c;被广泛的应用于负载均衡领域&#xff0c;如nginx和memcached都采用了一致性Hash来作为集群负载均衡的方案。 本文将介绍一致性Hash的基本思路&#xff0c;并讨论其…

计算机网络:局域网的数据链路层

✨✨ 欢迎大家来访Srlua的博文&#xff08;づ&#xffe3;3&#xffe3;&#xff09;づ╭❤&#xff5e;✨✨ &#x1f31f;&#x1f31f; 欢迎各位亲爱的读者&#xff0c;感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua小谢&#xff0c;在这里我会分享我的知识和经验。&am…

将 Elasticsearch 向量数据库引入到数据上的 Azure OpenAI 服务(预览)

作者&#xff1a;来自 Elastic Aditya Tripathi Microsoft 和 Elastic 很高兴地宣布&#xff0c;全球下载次数最多的向量数据库 Elasticsearch 是公共预览版中 Azure OpenAI Service On Your Data 官方支持的向量存储和检索增强搜索技术。 这项突破性的功能使你能够利用 GPT-4 …

Java数据结构-队列

目录 1. 队列概念2. 模拟实现队列2.1 链式队列2.2 循环队列 3. 双端队列4. 队列的应用4.1 用队列实现栈4.2 用栈实现队列 1. 队列概念 队列是一种只能在一端进行插入数据操作&#xff0c;另一端进行删除数据操作的数据结构&#xff0c;插入数据的叫队尾&#xff0c;删除数据的…

matlab——基于三维激光扫描点云的树冠体积计算方法

目录 一、算法原理1、原理概述2、参考文献二、代码实现三、结果展示本文由CSDN点云侠原创,原文链接。如果你不是在点云侠的博客中看到该文章,那么此处便是不要脸的爬虫与GPT。 一、算法原理 1、原理概述 针对树冠形状不规则,树冠体积难以测量和计算的问题,提出一种基于三…

第二十二章 Maven

一、Maven 1. Maven 简介 Maven 是一个项目管理工具&#xff0c;可以对 Java 项目进行自动化的构建和依赖管理。Maven 在美国是一个口语化的词语&#xff0c;代表专家、内行的意思&#xff0c;约等于北京话中的老炮儿。有老炮儿在身边&#xff0c;项目经理可谓得心应手。 项…

北京哪位医生治疗糖尿病比较好?

糖尿病是一种常见的慢性疾病&#xff0c;主要是由于胰岛素分泌不足或利用障碍导致的以高血糖为特征的代谢性疾病。持续的高血糖和长期的代谢紊乱可能导致全身组织器官&#xff0c;特别是眼、肾、心血管及神经系统的损害和功能障碍。 北京崇文门医院朱学敏主任在糖尿病领域有有着…

Go 源码之 gin 框架

Go 源码之 gin 框架 go源码之gin - Jxy 博客 一、总结 gin.New()初始化一个实例&#xff1a;gin.engine&#xff0c;该实例实现了http.Handler接口。实现了ServeHTTP方法 注册路由、注册中间件&#xff0c;调用addRoute将路由和中间件注册到 methodTree 前缀树&#xff08;节…

AcWing---转圈游戏---快速幂

太久没写快速幂了... 这是一道数学题orz&#xff0c;能看出来的话答案就是 &#xff0c;但是很大&#xff0c;同时还要mod n&#xff0c;直接用快速幂即可。 快速幂模版&#xff1a; long long int power(long long int a,long long int b,long long int mod){long long int r…