一、硬件电路:

1、引脚功能:

(1)A0-A2:决定不同设备的地址码:
(2)WP:写保护
二、通讯方式(IIC协议)
通讯方式与PCF8591相同,可参考以下文章:
蓝桥杯模块学习16——PCF8591(深夜学习——单片机)_佛科院深夜学习的博客-CSDN博客
1、设备地址:

由于我们使用的02型号的所以是2K的
2、按地址读取

先写入读取地址再进行读取
3、连续写或读操作:
我们只需在写或读完数据字节后发送应答信号(“1”),就能进行连续读或写,2K EEPROM最多可以连续读写8个字节
三、AT24C02实验:

1、代码思路:
定时器1,数码管——》读取AT24C02——》写入AT24C02——》满足实验要求
2、参考代码:
由于题目表达不清晰,我以为是一直循环:将数据+1、+2、+3,再往内存单元中写入,所以给自己增加难度了,如果你想挑战自己可以尝试一下,如果不想可以参考一下文章:
(5条消息) 【蓝桥杯单片机进阶强化-03】24C02存储器的基本原理与应用_小蜜蜂老师的博客-CSDN博客
(1)IIC代码:
#ifndef _IIC_H
#define _IIC_H
#include <STC15F2K60S2.H>
#include "intrins.h"
#define u8 unsigned char
#define u16 unsigned int
sbit SDA = P2^1;
sbit SCL = P2^0;
void IIC_Start(void);
void IIC_Stop(void);
bit IIC_WaitAck(void);
void IIC_SendAck(bit ackbit);
void IIC_SendByte(unsigned char byt);
unsigned char IIC_RecByte(void);
u8 AT24C02_Read_one(u8 adr);
void AT24C02_Write_one(u8 adr,u8 w_dat);
#endif
#include "iic.h"
#define DELAY_TIME 5
//
void IIC_Delay(unsigned char i)
{
do{_nop_();}
while(i--);
}
//
void IIC_Start(void)
{
SDA = 1;
SCL = 1;
IIC_Delay(DELAY_TIME);
SDA = 0;
IIC_Delay(DELAY_TIME);
SCL = 0;
}
//
void IIC_Stop(void)
{
SDA = 0;
SCL = 1;
IIC_Delay(DELAY_TIME);
SDA = 1;
IIC_Delay(DELAY_TIME);
}
//
void IIC_SendAck(bit ackbit)
{
SCL = 0;
SDA = ackbit;
IIC_Delay(DELAY_TIME);
SCL = 1;
IIC_Delay(DELAY_TIME);
SCL = 0;
SDA = 1;
IIC_Delay(DELAY_TIME);
}
//
bit IIC_WaitAck(void)
{
bit ackbit;
SCL = 1;
IIC_Delay(DELAY_TIME);
ackbit = SDA;
SCL = 0;
IIC_Delay(DELAY_TIME);
return ackbit;
}
//
void IIC_SendByte(unsigned char byt)
{
unsigned char i;
for(i=0; i<8; i++)
{
SCL = 0;
IIC_Delay(DELAY_TIME);
if(byt & 0x80) SDA = 1;
else SDA = 0;
IIC_Delay(DELAY_TIME);
SCL = 1;
byt <<= 1;
IIC_Delay(DELAY_TIME);
}
SCL = 0;
}
//
unsigned char IIC_RecByte(void)
{
unsigned char i, da;
for(i=0; i<8; i++)
{
SCL = 1;
IIC_Delay(DELAY_TIME);
da <<= 1;
if(SDA) da |= 1;
SCL = 0;
IIC_Delay(DELAY_TIME);
}
return da;
}
u8 AT24C02_Read_one(u8 adr)
{
u8 r_dat;
IIC_Start();
IIC_SendByte(0xa0);
IIC_WaitAck();
IIC_SendByte(adr);
IIC_WaitAck();
IIC_Start();
IIC_SendByte(0xa1);
IIC_WaitAck();
r_dat = IIC_RecByte();
IIC_SendAck(1);
IIC_Stop();
return r_dat;
}
void AT24C02_Write_one(u8 adr,u8 w_dat)
{
IIC_Start();
IIC_SendByte(0xa0);
IIC_WaitAck();
IIC_SendByte(adr);
IIC_WaitAck();
IIC_SendByte(w_dat);
IIC_WaitAck();
IIC_Stop();
}
(2)主函数:(我使用了一些非抢占式分配的思路,如果看不懂可以就看上面的那篇文章就行)
#include <STC15F2K60S2.H>
#include <stdio.H>
#include "iic.h"
#define u8 unsigned char
#define u16 unsigned int
code unsigned char Seg_Table[] =
{
0xc0, //0
0xf9, //1
0xa4, //2
0xb0, //3
0x99, //4
0x92, //5
0x82, //6
0xf8, //7
0x80, //8
0x90, //9
0x88, //A
0x83, //b
0xc6, //C
0xa1, //d
0x86, //E
0x8e //F
};
//数码管
u8 COD[8],COT[9],PSI;
u16 seg_delay;
//AT24C02
u8 at24_dat[5],show_dat[3];
u16 cout_11ms;
u16 AT24_dealy[3]={0,1,1};
u8 show_delay;
u16 ms_count;
void Close_All();
void Timer1_Init(void);
void SEG_Rroc();
void AT24C02_Proc();
void SEG_Show_Rroc();
void AT24C02_Read_Proc();
void AT24C02_Plus_Proc();
void AT24C02_Write_Proc();
void main()
{
Close_All();
Timer1_Init();
while(1)
{
SEG_Rroc();
SEG_Show_Rroc();
AT24C02_Read_Proc();
AT24C02_Plus_Proc();
AT24C02_Write_Proc();
}
}
/**************定时器******************/
void SEG_Show(u8 COD,u8 PSI);
void Timer1_Isr(void) interrupt 3
{
ms_count++;
if(ms_count == seg_delay) seg_delay = 0;
if(ms_count % 2 == 0) show_delay = 0;
if(ms_count == AT24_dealy[0]) AT24_dealy[0] = 0;
if(ms_count == AT24_dealy[1]) AT24_dealy[1] = 0;
if(ms_count == AT24_dealy[2]) AT24_dealy[2] = 0;
if(ms_count == 1000) ms_count = 0;
}
void Timer1_Init(void) //1毫秒@12.000MHz
{
AUXR &= 0xBF; //定时器时钟12T模式
TMOD &= 0x0F; //设置定时器模式
TL1 = 0x18; //设置定时初始值
TH1 = 0xFC; //设置定时初始值
TF1 = 0; //清除TF1标志
TR1 = 1; //定时器1开始计时
ET1 = 1; //使能定时器1中断
EA = 1;
}
/*************数码管*******************/
void SEG_TSL(u8* input,u8* output)
{
u8 i;
for(i=0;i<8;i++)
{
switch(input[i])
{
case '0':output[i] = Seg_Table[0];break;
case '1':output[i] = Seg_Table[1];break;
case '2':output[i] = Seg_Table[2];break;
case '3':output[i] = Seg_Table[3];break;
case '4':output[i] = Seg_Table[4];break;
case '5':output[i] = Seg_Table[5];break;
case '6':output[i] = Seg_Table[6];break;
case '7':output[i] = Seg_Table[7];break;
case '8':output[i] = Seg_Table[8];break;
case '9':output[i] = Seg_Table[9];break;
case '-':output[i] = ~0x40;break;
default:output[i] = 0xff;
}
}
}
void SEG_Show(u8 COD,u8 PSI)
{
//消隐
P0 = 0xff;
P2 = P2 & 0x1f | (0x70<<1);
P2 &= 0x1f;
//位选
P0 = 0x01<<PSI;
P2 = P2 & 0x1f | (0x60<<1);
P2 &= 0x1f;
//段选
P0 = COD;
P2 = P2 & 0x1f | (0x70<<1);
P2 &= 0x1f;
}
/**************************************/
/*
关闭无关设备
*/
void Close_All()
{
//关闭蜂鸣器和继电器
P0 = 0x00;
P2 = P2 & 0x1f | (0x50<<1);
P2 &= 0x1f;
//关闭LED
P0 = 0xff;
P2 = P2 & 0x1f | (0x40<<1);
P2 &= 0x1f;
}
void SEG_Rroc()
{
if(seg_delay)return;
seg_delay = 997;
sprintf(COT,"%02u-%02u-%02u",(u16)show_dat[0],(u16)show_dat[1],(u16)show_dat[2]);
SEG_TSL(COT,COD);
}
void SEG_Show_Rroc()
{
if(show_delay)return;
show_delay = 1;
SEG_Show(COD[PSI],PSI);
if(PSI++ == 7)PSI = 0;
}
void AT24C02_Read_Proc()
{
static r_num=0;
if(AT24_dealy[0])return;
AT24_dealy[0] = 499;
AT24_dealy[0] += 10*(r_num+1);
EA = 0;
at24_dat[r_num] = AT24C02_Read_one(0x01+2*r_num);
show_dat[r_num] = at24_dat[r_num];
EA = 1;
if(r_num++ == 2)r_num = 0;
}
void AT24C02_Plus_Proc()
{
if(AT24_dealy[1])return;
AT24_dealy[1] = 599;
at24_dat[0] +=1;
at24_dat[1] +=2;
at24_dat[2] +=3;
if(at24_dat[0] > 10)
at24_dat[0] = 0;
if(at24_dat[1] > 20)
at24_dat[1] = 0;
if(at24_dat[2] > 30)
at24_dat[2] = 0;
}
void AT24C02_Write_Proc()
{
static w_num=0;
if(AT24_dealy[2])return;
AT24_dealy[2] = 699;
AT24_dealy[2] += 4*(w_num+1);
EA = 0;
AT24C02_Write_one(0x01+2*w_num,at24_dat[w_num]);
EA = 1;
if(w_num++ == 2)w_num = 0;
}