当前位置:   article > 正文

(十一)51单片机——用AT24C02实现存储秒表数据(附成果展示)_利用at24c02完成具有记忆功能的秒表发生器

利用at24c02完成具有记忆功能的秒表发生器

目录

存储器

RAM

ROM

存储器简化模型

AT24C02介绍

 引脚及应用电路

 内部结构框图

I2C总线

I2C总线介绍

I2C电路规范

I2C时序结构

起始条件

终止条件

发送一个字节 

接受一个字节 

发送应答 

接收应答

I2C数据帧

AT24C02数据帧

字节写

随机读

代码部分

遇到的问题 

代码 

硬件


    

        今天我们来介绍一下AT24C02,首先呢,它是一种可以实现掉电不丢失的存储器,可用于保存单片机运行时想要永久保存的数据信息,在介绍AT24C02之前,我们先来介绍一下存储器!

存储器

        先来简单介绍一下RAM(随机存储器)以及ROM(只读存储器)的优缺点吧!

优点缺点
RAM储存速度快掉电丢失
ROM存储速度慢掉电不丢失

RAM

        RAM主要分为SRAM(静态RAM)和DRAM(动态RAM),SRAM主要用于电脑CPU以及我们的单片机CPU;而DRAM主要用在电脑内存条以及手机的运行内存,因为电容器会掉电,所以需要不断进行扫描。

组成优点缺点
SRAM触发器存储速度较快容量小,成本较高
DRAM电容存储速度较慢容量大,成本较低

ROM

        ROM主要分为Mask ROM(掩膜ROM),PROM(可编程ROM),EPROM(可擦除可编程ROM ),E2PROM (电可擦除可编程ROM ),这四个是一家的,还有Flash(闪存),硬盘、软盘、光盘等,其中Flash目前使用十分广泛,基本上打败了ROM一家。

特点
Mask ROM只能读
PROM可以写,但只能一次
EPROM可以写多次,但要紫外线照射30分钟
E2PROM可以写多次,并且只要几毫秒即可
Flash与E2PROM类似,但集成度更高
硬盘、软盘、光盘等软盘和光盘目前见的比较少了

存储器简化模型

        这个地方涉及到了数电的知识,稍后会出数电寄存器一章的笔记,目前我们只能简单的讲解一下。左边是地址总线,下面是数据总线,首先我们选择地址总线,比如像赋值10000000,相当于打开了第一行,之后选择连接的结点(之前都没有连接上),将其连上,Mask ROM使用的方法是一个二极管(这么做的原因是防止电流经过上面的节点导致数据混乱),而PROM使用了两个二极管(一个二极管和保险丝),但是其中一个二极管(保险丝)比较容易击穿,当给高电压的时候,蓝色电容(保险丝熔断)击穿,实现数据写入。这也是我们“烧录”的由来,然后我们现在的就是属于给电之后会恢复,实现反复写入,具体是怎么样的,我们在稍后的数电笔记中进行详细的介绍。 

AT24C02介绍

          接下来我们来简单介绍一下AT24C02吧! 

  • AT24C02是一种可以实现掉电不丢失的存储器,可用于保存单片机运行时想要永久保存的数据信息
  • 存储介质:E2PROM
  • 通讯接口:I2C总线
  • 容量:256字节

 引脚及应用电路

 内部结构框图

接下来我们来简单的介绍一下内部结构,我们从每个部分进行讲解!

  1. 第一个就是我们刚刚介绍的存储器简化模型那样,网状结构
  2. 第二个是一个译码器,用于输入地址
  3. 第三个是输入输出端,通过Y DEC将数据输出
  4. 第四个也是译码器,用来帮助MUX输出数据,然后就直接输出数据
  5. 第五个是用来擦除数据用的
  6. 第六个是用来设置地址的,里面有个寄存器是用来存储地址的,每写入和读出寄存器自动加一,读出不指定地址,默认拿出寄存器的地址
  7. 第七个是开始结束逻辑
  8. 第八个是一个地址比较器
  9. 第九个是一个控制串行逻辑

I2C总线

  • I2C总线介绍

  • I2C总线(Inter IC BUS)是由Philips公司开发的一种通用数据总线
  • 两根通信线:SCL(Serial Clock)、SDA(Serial Data)
  • 同步、半双工,带数据应答
  • 通用的I2C总线,可以使各种设备的通信标准统一,对于厂家来说,使用成熟的方案可以缩短芯片设计周期、提高稳定性,对于应用者来说,使用通用的通信协议可以避免学习各种各样的自定义协议,降低了学习和应用的难度

I2C电路规范

其中一个IC的内部结构

我们来抽象一下I2C的通信方式

通信规则:

1、杠子在上方代表1,下方代表0

2、每个人只能拉杆子或者松开手

3、每个人需要地址进行通信

I2C时序结构

        接下来我们来介绍一下六个时序结构,只要集齐了这六个时序结构,就可以召唤数据帧了!

起始条件

        起始条件:SCL高电平期间,SDA从高电平切换到低电平(相当于告诉大家我要发送信息了) 

终止条件

        终止条件:SCL高电平期间,SDA从低电平切换到高电平(相当于告诉大家我要停止了)

发送一个字节 

        发送一个字节:SCL低电平期间,主机将数据位依次放到SDA线上(高位在前),然后拉高SCL,从机将在SCL高电平期间读取数据位,所以SCL高电平期间SDA不允许有数据变化,依次循环上述过程8次,即可发送一个字节

接受一个字节 

        接收一个字节:SCL低电平期间,从机将数据位依次放到SDA线上(高位在前),然后拉高SCL,主机将在SCL高电平期间读取数据位,所以SCL高电平期间SDA不允许有数据变化,依次循环上述过程8次,即可接收一个字节(主机在接收之前,需要释放SDA)

发送应答 

        发送应答(SA):在接收完一个字节之后,主机在下一个时钟发送一位数据,数据0表示应答,数据1表示非应答(相当于一个回应)

接收应答

        接收应答(RA):在发送完一个字节之后,主机在下一个时钟接收一位数据,判断从机是否应答,数据0表示应答,数据1表示非应答(主机在接收之前,需要释放SDA) 

I2C数据帧

        I2C数据帧其实就是上面六个部分拼合在一起,把数据帧拆分开来看,就比较好理解了。

发送一帧数据 

        相当于老师在讲课,我们给老师回复

接收一帧数据

        相当于老师叫人回答问题 

复合格式

像是一个完整的回答过程,老师提出问题,学生回答 

AT24C02数据帧

        AT24C02数据帧,其实不止这几个,但我们在这里就简单介绍一下这两种!

字节写

字节写:在WORD ADDRESS处写入数据DATA

随机读

随机读:读出在WORD ADDRESS处的数据DATA

        AT24C02的固定地址为1010,可配置地址本开发板上为000,所以SLAVE ADDRESS+W为0xA0,SLAVE ADDRESS+R为0xA1

代码部分

        这节内容的代码有一点点面向对象的思想,因为AT24C02的时序帧是根据I2C的六个时序结构拼接而成,所以AT24C02时序帧只需要将他们拼装起来,有点类似与接口和继承的味道,好了,我们先将代码给出!

  1. // I2C.c
  2. #include <REGX52.H>
  3. // 在引脚部分介绍过了,SCL是P21,SDA是P20;
  4. sbit I2C_SCL=P2^1;
  5. sbit I2C_SDA=P2^0;
  6. // 单片机比较慢无需delay,这个就是按照时序图来的,按图来就行
  7. /**
  8. * @brief I2C开始
  9. * @param 无
  10. * @retval 无
  11. */
  12. void I2C_Start(void)
  13. {
  14. // 可以理解为初始化,确保一定为高电平
  15. I2C_SDA = 1;
  16. I2C_SCL = 1;
  17. // 按照时序图可得,先SDA为0,再SCL为0
  18. I2C_SDA = 0;
  19. I2C_SCL = 0;
  20. }
  21. /**
  22. * @brief I2C停止
  23. * @param 无
  24. * @retval 无
  25. */
  26. void I2C_Stop(void)
  27. {
  28. I2C_SDA = 0;
  29. I2C_SCL = 1;
  30. I2C_SDA = 1;
  31. }
  32. /**
  33. * @brief I2C发送一个字节
  34. * @param Byte 要发送的字节
  35. * @retval 无
  36. */
  37. void I2C_SendByte(unsigned char Byte)
  38. {
  39. unsigned char i;
  40. for(i = 0;i < 8;i++){
  41. I2C_SDA = Byte & (0x80 >> i);
  42. I2C_SCL = 1;
  43. I2C_SCL = 0;
  44. }
  45. }
  46. /**
  47. * @brief I2C接收一个字节
  48. * @param 无
  49. * @retval 接收到的一个字节数据
  50. */
  51. unsigned char I2C_ReceiveByte(void)
  52. {
  53. unsigned char i,Byte = 0x00;
  54. // 释放SDA
  55. I2C_SDA = 1;
  56. for(i = 0; i < 8; i++)
  57. {
  58. I2C_SCL=1;
  59. if(I2C_SDA){Byte|=(0x80>>i);}
  60. I2C_SCL=0;
  61. }
  62. return Byte;
  63. }
  64. /**
  65. * @brief I2C发送应答
  66. * @param AckBit 应答位,0为应答,1为非应答
  67. * @retval 无
  68. */
  69. void I2C_SendAck(unsigned char AckBit)
  70. {
  71. I2C_SDA = AckBit;
  72. I2C_SCL = 1;
  73. I2C_SCL = 0;
  74. }
  75. /**
  76. * @brief I2C接收应答位
  77. * @param 无
  78. * @retval 接收到的应答位,0为应答,1为非应答
  79. */
  80. unsigned char I2C_ReceiveAck(void)
  81. {
  82. unsigned char AckBit;
  83. // 释放SDA
  84. I2C_SDA = 1;
  85. I2C_SCL = 1;
  86. AckBit = I2C_SDA;
  87. I2C_SCL = 0;
  88. return AckBit;
  89. }

  1. // AT24C02.c
  2. #include <REGX52.H>
  3. #include "I2C.h"
  4. // SLAVE ADDRESS+W为0xA0,SLAVE ADDRESS+R为0xA1
  5. #define AT24C02_ADDRESS_READ 0xA0
  6. #define AT24C02_ADDRESS_WRITE 0xA1
  7. /**
  8. * @brief AT24C02写入一个字节
  9. * @param WordAddress 要写入字节的地址
  10. * @param Data 要写入的数据
  11. * @retval 无
  12. */
  13. void AT24C02_WriteByte(unsigned char WordAddress,Data)
  14. {
  15. I2C_Start();
  16. I2C_SendByte(AT24C02_ADDRESS_READ);
  17. I2C_ReceiveAck();
  18. I2C_SendByte(WordAddress);
  19. I2C_ReceiveAck();
  20. I2C_SendByte(Data);
  21. I2C_ReceiveAck();
  22. I2C_Stop();
  23. }
  24. /**
  25. * @brief AT24C02读取一个字节
  26. * @param WordAddress 要读出字节的地址
  27. * @retval 读出的数据
  28. */
  29. unsigned char AT24C02_ReadByte(unsigned char WordAddress)
  30. {
  31. unsigned char Data;
  32. I2C_Start();
  33. I2C_SendByte(AT24C02_ADDRESS_READ);
  34. I2C_ReceiveAck();
  35. I2C_SendByte(WordAddress);
  36. I2C_ReceiveAck();
  37. I2C_Start();
  38. // 读地址
  39. I2C_SendByte(AT24C02_ADDRESS_WRITE);
  40. I2C_ReceiveAck();
  41. Data=I2C_ReceiveByte();
  42. I2C_SendAck(1);
  43. I2C_Stop();
  44. return Data;
  45. }

接下来我们使用这些代码实现一个数据储存器,代码如下所示:

  1. //main.c
  2. #include <REGX52.H>
  3. #include "LCD1602.h"
  4. #include "Key.h"
  5. #include "Delay.h"
  6. #include "AT24C02.h"
  7. void main(){
  8. unsigned char KeyNum;
  9. unsigned int Num;
  10. LCD_Init();
  11. LCD_ShowNum(1,1,0,5);
  12. while(1){
  13. KeyNum = Key();
  14. if(KeyNum == 1)
  15. {
  16. Num++;
  17. LCD_ShowNum(1,1,Num,5);
  18. }
  19. if(KeyNum == 2)
  20. {
  21. Num--;
  22. LCD_ShowNum(1,1,Num,5);
  23. }
  24. if(KeyNum == 3)
  25. {
  26. AT24C02_WriteByte(0,Num%256);// 取低八位
  27. Delay(5);// 因为读周期为5毫秒,如果不延时,将读不出结果
  28. AT24C02_WriteByte(1,Num/256);// 取高八位
  29. Delay(5);
  30. LCD_ShowString(2,1,"Write OK");
  31. Delay(1000);
  32. LCD_ShowString(2,1," ");
  33. }
  34. if(KeyNum == 4)
  35. {
  36. Num = AT24C02_ReadByte(0);// 读低八位
  37. Num |= AT24C02_ReadByte(1) << 8;// 读高八位
  38. LCD_ShowNum(1,1,Num,5);
  39. LCD_ShowString(2,1,"Read OK");
  40. Delay(1000);
  41. LCD_ShowString(2,1," ");
  42. }
  43. }
  44. }

运行效果如下所示:

AT24C02存储


        接下来,我们将会改进之前动态数码管的实现,使用定时器来扫描,然后实现一个具有记忆功能的秒表。

思路 

        我们使用定时器来扫描按键以及数码管,所以按键以及数码管都需要用到定时器的功能,具体内容如下所示:

        但我们只有一个中断函数,这样很容易出错,而且不能达到目的,并且代码耦合性过高,所以我们采用另一种方式,如下所示。就是将定时函数放到主函数里面去,再每隔一段时间调用各个部分的函数以达到目的,好了,接下来我们看看代码是如何实现的吧!

  1. //Nixie.c
  2. #include <REGX52.H>
  3. #include "Delay.h"
  4. // 存放数码管显示缓存区
  5. unsigned char Nixie_Buf[9] = {0, 10, 10, 10, 10, 10, 10, 10, 10};
  6. /**
  7. * @brief 设置显示缓存区
  8. * @param Location 要设置的位置,范围:1~8
  9. * @param Number 要设置的数字,范围:段码表索引范围
  10. * @retval 无
  11. */
  12. void Nixie_SetBuf(unsigned char Location, unsigned char Number)
  13. {
  14. Nixie_Buf[Location] = Number;
  15. }
  16. /**
  17. * @brief 数码管扫描显示
  18. * @param Location 要显示的位置,范围:1~8
  19. * @param Number 要显示的数字,范围:段码表索引范围
  20. * @retval 无
  21. */
  22. void Nixie_Scan(unsigned char Location, unsigned char Number)
  23. {
  24. unsigned char NixieTable[] = {0x3F, 0x06, 0x5B, 0x4F, 0x66, 0x6D, 0x7D, 0x07, 0x7F, 0x6F, 0x00,0x40};
  25. P0 = 0x00;//段码清0,消影
  26. switch(Location)//位码输出
  27. {
  28. case 1:P2_4=1;P2_3=1;P2_2=1;break;
  29. case 2:P2_4=1;P2_3=1;P2_2=0;break;
  30. case 3:P2_4=1;P2_3=0;P2_2=1;break;
  31. case 4:P2_4=1;P2_3=0;P2_2=0;break;
  32. case 5:P2_4=0;P2_3=1;P2_2=1;break;
  33. case 6:P2_4=0;P2_3=1;P2_2=0;break;
  34. case 7:P2_4=0;P2_3=0;P2_2=1;break;
  35. case 8:P2_4=0;P2_3=0;P2_2=0;break;
  36. }
  37. P0 = NixieTable[Number];//段码输出
  38. }
  39. // 数码管扫描
  40. void Nixie_Loop(void)
  41. {
  42. static unsigned char i = 1;
  43. Nixie_Scan(i,Nixie_Buf[i]);
  44. i++;
  45. if(i>=9){i = 1;}
  46. }

  1. //Key.c
  2. #include <REGX52.H>
  3. #include "Delay.h"
  4. unsigned char Key_KeyNumber;
  5. /**
  6. * @brief 获取独立按键键码
  7. * @param 无
  8. * @retval 按下按键的键码,范围:0~4,无按键按下时返回值为0
  9. */
  10. unsigned char Key_GetState()
  11. {
  12. unsigned char KeyNumber=0;
  13. if(P3_1==0){KeyNumber=1;}
  14. if(P3_0==0){KeyNumber=2;}
  15. if(P3_2==0){KeyNumber=3;}
  16. if(P3_3==0){KeyNumber=4;}
  17. return KeyNumber;
  18. }
  19. // 循环调用函数,用来判断哪个键被按下
  20. void Key_Loop(void)
  21. {
  22. static unsigned char NowState, LastState;
  23. LastState = NowState;
  24. NowState = Key_GetState();
  25. // 上一次按键1被按下,然后按键松开,完成一次按键识别
  26. if(LastState == 1 && NowState == 0)
  27. {
  28. Key_KeyNumber = 1;
  29. }
  30. if(LastState == 2 && NowState == 0)
  31. {
  32. Key_KeyNumber = 2;
  33. }
  34. if(LastState == 3 && NowState == 0)
  35. {
  36. Key_KeyNumber = 3;
  37. }
  38. if(LastState == 4 && NowState == 0)
  39. {
  40. Key_KeyNumber = 4;
  41. }
  42. }
  43. unsigned char Key(void){
  44. unsigned char Temp = 0;
  45. Temp = Key_KeyNumber;
  46. // 将Key_KeyNumber置0,因为Key_KeyNumber不会刷新
  47. Key_KeyNumber = 0;
  48. return Temp;
  49. }

  1. //main.c
  2. #include <REGX52.H>
  3. #include "Time0.h"
  4. #include "Key.h"
  5. #include "Nixie.h"
  6. #include "Delay.h"
  7. #include "AT24C02.h"
  8. unsigned char KeyNum,Min, Sec, MiniSec,RunFlag;
  9. void main(){
  10. Timer0_Init();
  11. while(1){
  12. KeyNum = Key();
  13. if(KeyNum == 1)//K1按键按下
  14. {
  15. RunFlag =!RunFlag;//启动标志位翻转
  16. }
  17. if(KeyNum == 2)//K2按键按下
  18. {
  19. //分秒清0
  20. Min = 0;
  21. Sec = 0;
  22. MiniSec = 0;
  23. }
  24. if(KeyNum == 3)//K3按键按下
  25. {
  26. //将分秒写入AT24C02
  27. AT24C02_WriteByte(0,Min);
  28. Delay(5);
  29. AT24C02_WriteByte(1,Sec);
  30. Delay(5);
  31. AT24C02_WriteByte(2,MiniSec);
  32. Delay(5);
  33. }
  34. if(KeyNum == 4)//K4按键按下
  35. {
  36. //读出AT24C02数据
  37. Min = AT24C02_ReadByte(0);
  38. Sec = AT24C02_ReadByte(1);
  39. MiniSec = AT24C02_ReadByte(2);
  40. }
  41. //设置显示缓存,显示数据
  42. Nixie_SetBuf(1,Min/10);
  43. Nixie_SetBuf(2,Min%10);
  44. Nixie_SetBuf(3,11);
  45. Nixie_SetBuf(4,Sec/10);
  46. Nixie_SetBuf(5,Sec%10);
  47. Nixie_SetBuf(6,11);
  48. Nixie_SetBuf(7,MiniSec/10);
  49. Nixie_SetBuf(8,MiniSec%10);
  50. }
  51. }
  52. void Sec_Loop(void)
  53. {
  54. if(RunFlag){
  55. MiniSec++;
  56. if(MiniSec >= 100)
  57. {
  58. MiniSec = 0;
  59. Sec++;
  60. if(Sec>=60)
  61. {
  62. Sec = 0;
  63. Min++;
  64. if(Min>=60)
  65. {
  66. Min = 0;
  67. }
  68. }
  69. }
  70. }
  71. }
  72. // 定时函数里面的函数千万不能有延时,因为每一毫秒都要进来一次,会卡住
  73. void Timer0_Routine() interrupt 1
  74. {
  75. static unsigned int T0Count,T0Count1,T0Count2;
  76. TL0 = 0x66; //设置定时初始值
  77. TH0 = 0xFC;
  78. // 20ms一次按键扫描
  79. T0Count++;
  80. if(T0Count >= 20){
  81. T0Count = 0;
  82. Key_Loop();
  83. }
  84. // 2ms一次数码管扫描
  85. T0Count1++;
  86. if(T0Count1 >= 2){
  87. T0Count1 = 0;
  88. Nixie_Loop();
  89. }
  90. //10ms调用一次数秒表驱动函数
  91. T0Count2++;
  92. if(T0Count2 >= 10){
  93. T0Count2 = 0;
  94. Sec_Loop();
  95. }
  96. }

运行结果如下所示:

秒表

        好了,我们关于AT24C02的知识点就先介绍到这里,接下来还会继续分享关于51单片机的知识!

遇到的问题 

代码 

        编写程序的整个过程中,不小心在中途把Keil的一些启动文件给删了,然后花了一小时重新下载;而且因为这次涉及的模块比较多,而且采用了定时器扫描的思路,导致编写代码过程比较艰难,但不断调试和细心纠错,还是能慢慢找到问题的。 

硬件

        第一次使用单片机的时候就烧坏过CPU,然后点阵屏也出过问题,然后这次编写代码过程中LCD1602以及数码管也出了问题,一直没有办法,都准备换一个单片机了,最后移动了一下CPU,就恢复了,硬件出问题,真的太痛苦了。 

声明:本文内容由网友自发贡献,转载请注明出处:【wpsshop】
推荐阅读
相关标签
  

闽ICP备14008679号