蓝桥杯模块学习17——AT24C02存储器(深夜学习——单片机)

一、硬件电路:

1、引脚功能:

(1)A0-A2:决定不同设备的地址码:

(2)WP:写保护

二、通讯方式(IIC协议)

通讯方式与PCF8591相同,可参考以下文章:

蓝桥杯模块学习16——PCF8591(深夜学习——单片机)_佛科院深夜学习的博客-CSDN博客

1、设备地址:

由于我们使用的02型号的所以是2K的

2、按地址读取

先写入读取地址再进行读取

3、连续写或读操作:

我们只需在写或读完数据字节后发送应答信号(“1”),就能进行连续读或写,2K EEPROM最多可以连续读写8个字节

  1. 注意:

IIC写一次时间大概要5ms,不能刚进行写操作后就进行读操作

三、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;
}
  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是使用Arduino UNO和AT24C02存储器进行读写实验,并通过液晶显示读取的数据的示例代码。 首先,需要连接AT24C02存储器和液晶显示器到Arduino UNO上。AT24C02使用I2C协议连接到Arduino UNO的A4和A5引脚。液晶显示器则连接到数字引脚2、3、4、5、11和12。 示例代码如下: ```C++ #include <Wire.h> #include <LiquidCrystal.h> LiquidCrystal lcd(2, 3, 4, 5, 11, 12); // 设置液晶显示器引脚 #define EEPROM_ADDR 0x50 // AT24C02存储器地址 void setup() { Serial.begin(9600); Wire.begin(); lcd.begin(16, 2); // 初始化液晶显示器 lcd.setCursor(0, 0); lcd.print("AT24C02 Test"); } void loop() { int value = random(256); // 随机生成一个0到255的整数 writeEEPROM(0, value); // 写入AT24C02存储器 int readValue = readEEPROM(0); // 从AT24C02存储器读取数据 lcd.setCursor(0, 1); lcd.print("Value: "); lcd.print(readValue); delay(5000); // 延迟5秒 } void writeEEPROM(int address, int data) { Wire.beginTransmission(EEPROM_ADDR); Wire.write((int)(address >> 8)); Wire.write((int)(address & 0xFF)); Wire.write(data); Wire.endTransmission(); delay(5); } int readEEPROM(int address) { int value = 0; Wire.beginTransmission(EEPROM_ADDR); Wire.write((int)(address >> 8)); Wire.write((int)(address & 0xFF)); Wire.endTransmission(); Wire.requestFrom(EEPROM_ADDR, 1); if (Wire.available()) { value = Wire.read(); } return value; } ``` 在setup函数中,初始化串口和液晶显示器,并显示一个欢迎消息。在loop函数中,生成一个随机整数,并将其写入AT24C02存储器中。然后,从存储器中读取数据并在液晶显示器上显示。最后,延迟5秒以等待下一个循环。 writeEEPROM函数用于将数据写入AT24C02存储器中。它接受两个参数:地址和要写入的数据。它首先发送存储器地址,然后发送要写入的数据。最后,它结束传输并延迟5毫秒以确保数据被写入存储器。 readEEPROM函数用于从AT24C02存储器中读取数据。它接受一个参数:地址。它首先发送存储器地址,然后请求一个字节的数据。如果数据可用,则将其读取并返回。 在液晶显示器上,第一行显示欢迎消息,第二行显示从存储器中读取的数据。 注意:在使用AT24C02存储器之前,请确保将其正确连接到Arduino UNO,并确保I2C地址正确。如果您遇到问题,请参阅AT24C02数据表以获取更多信息。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值