蓝桥杯单片机之模块代码《AT24C02页写入》

过往历程

历程1:秒表

历程2:按键显示时钟

历程3:列矩阵按键显示时钟

历程4:行矩阵按键显示时钟

历程5:新DS1302

历程6:小数点精确后两位ds18b20

历程7:35定时器测量频率

历程8:AT24C02

历程9:AT24C02秒表

历程10:AD

历程11:超声波

历程12:多样点灯方式



AT24C02介绍

AT24C02是一种可以实现掉电不丢失的存储器,可用于保存单片机运行时想要永久保存的数据信息

  • 存储介质:E2PROM
  • 通讯接口:I2C总线
  • 容量:256字节
    在这里插入图片描述

在蓝桥杯中,他的电路图是这样的
在这里插入图片描述

前期准备

1:本文使用的头文件是由stc生成
在这里插入图片描述
点击保存文件之后,保存至自己文件目录下方即可
在这里插入图片描述
2:本文使用的都是官方提供的底层驱动代码
在这里插入图片描述

驱动处理

AD转换/DA转换

官方提供的程序

//c文件
#include "iic.h"

#define DELAY_TIME 5

//I2C总线内部延时函数
void IIC_Delay(unsigned char i)
{
    do{_nop_();}
    while(i--);        
}

//I2C总线启动信号
void IIC_Start(void)
{
    SDA = 1;
    SCL = 1;
    IIC_Delay(DELAY_TIME);
    SDA = 0;
    IIC_Delay(DELAY_TIME);
    SCL = 0;	
}

//I2C总线停止信号
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;
}

//I2C总线发送一个字节数据
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;  
}

//I2C总线接收一个字节数据
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;    
}

//头文件
#ifndef _IIC_H
#define _IIC_H

#include "reg52.h"
#include "intrins.h"

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); 

#endif

自己需要更改的地方

1:同样这个是与ds1302一样,只需要添加上中断即可

#include "iic.h"

#define DELAY_TIME 5

//I2C总线内部延时函数
void IIC_Delay(unsigned char i)
{
    do{_nop_();}
    while(i--);        
}

//I2C总线启动信号
void IIC_Start(void)
{
    SDA = 1;
    SCL = 1;
    IIC_Delay(DELAY_TIME);
    SDA = 0;
    IIC_Delay(DELAY_TIME);
    SCL = 0;	
}

//I2C总线停止信号
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;
}

//I2C总线发送一个字节数据
void IIC_SendByte(unsigned char byt)
{
    unsigned char i;

    for(i=0; i<8; i++)
    {
		EA=0;							//关闭中断
        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);
			EA=1;					//开启中断
    }
    SCL  = 0;  
}

//I2C总线接收一个字节数据
unsigned char IIC_RecByte(void)
{
    unsigned char i, da;
    for(i=0; i<8; i++)
    {   
			EA=0;			//关闭中断
    	SCL = 1;	
	IIC_Delay(DELAY_TIME);
	da <<= 1;
	if(SDA) da |= 1;
	SCL = 0;
	IIC_Delay(DELAY_TIME);
			EA=1;		//开启中断
    }
    return da;    
}


本期代码

#include "stc15f2k.h"
#include "iic.h"
#include "intrins.h"

sbit s7=P3^0;
sbit s6=P3^1;

char dula[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0xbf,0xff};
char disdula[]={0,0,0,0,0,0,0,0};

char aa[]={0,0,0,0};

unsigned int mode=1,wei=0;
bit flag_display;

void buzz()
{
	P2=((P2&0X1F)|0XA0);
	P0=0X00;
	P2&=0X1F;
	
	P2=((P2&0X1F)|0X80);
	P0=0XFF;
	P2&=0X1F;
}

void delay(int x)
{
	int i;
	while(x--)
		for(i=0;i<5*123;i++);
}

void din(void)		//1??@11.0592MHz
{
	AUXR &= 0xBF;		//?????12T??
	TMOD &= 0x0F;		//???????
	TL1 = 0x66;		//??????
	TH1 = 0xFC;		//??????
	TF1 = 0;		//??TF1??
	TR1 = 1;		//???1????
	EA=1;
	ET1=1;
}

void display()
{
	unsigned int diswela;

	
	P2=((P2&0X1F)|0Xe0);
	P0=0XFF;
	P2&=0X1F;
	
	P2=((P2&0X1F)|0Xc0);
	P0=1<<diswela;
	P2&=0X1F;
	
	P2=((P2&0X1F)|0Xe0);
	P0=dula[disdula[diswela]];
	P2&=0X1F;
	
	if(++diswela>=8)diswela=0;
}

void display_at24c02()
{
	disdula[0]=11;
	disdula[1]=mode;
	disdula[2]=11;
	disdula[3]=11;
	disdula[4]=11;
	disdula[5]=11;
	disdula[6]=aa[mode-1]/10%10;
	disdula[7]=aa[mode-1]/1%10;
}


char anjian(void)
{
	static int key_up=1;
	unsigned int keyscan=0;
	if(key_up&(s7==0||s6==0))
	{
		delay(10);
		key_up=0;
		if(s7==0)keyscan=7;
		else if(s6==0)keyscan=6;
	}else if(s7==1&&s6==1)key_up=1;
	return keyscan;
}


void at24c02_write(int add,char *date,int num)
{
	IIC_Start();
	IIC_SendByte(0xa0);
	IIC_WaitAck();
	IIC_SendByte(add);
	IIC_WaitAck();
	while(num--)
	{
	IIC_SendByte(*date++);
	IIC_WaitAck();
	delay(3);
	}
	IIC_Stop();
}

void at24c02_read(int add,char *date,int num)
{
	IIC_Start();
	IIC_SendByte(0xa0);
	IIC_WaitAck();
	IIC_SendByte(add);
	IIC_WaitAck();
	
	IIC_Start();
	IIC_SendByte(0xa1);
	IIC_WaitAck();
	while(num--)
	{
	*date++=IIC_RecByte();
	if(num)
	IIC_SendAck(0);
	else 
	IIC_SendAck(1);
	}
	IIC_Stop();
}

void main()
{
	unsigned int key;
	buzz();	
	at24c02_read(0x21,aa,4);
	din();
	while(1)
	{
		display_at24c02();
		
		if(flag_display==1)
		{
			flag_display=0;
			key=anjian();
			
			switch(key)
			{
				case 7:
					if(wei==0)
					{
						if(++mode>4)mode=1;
					}
					else if(wei==1)
					{
						aa[mode-1]+=1;
						if(aa[mode-1]>20)aa[mode-1]=0;at24c02_write(0x21,aa,4);
					}
				break;
				case 6:
					if(wei==0)wei=1;
				else wei=0;
				break;
			}
		}
	}
}


unsigned int ms;            //²»ÖªµÀΪʲôÔÚÀïÃ涨Òå¾Í²»ÐÐ
void qa() interrupt 3
{

	ms++;
	display();
	if(ms>=1500)ms=0;
	if(!(ms%100))flag_display=1;
}
		
	

省赛代码传送门

第三届蓝桥杯单片机省赛
第四届蓝桥杯单片机省赛
第五届蓝桥杯单片机省赛
第六届蓝桥杯单片机省赛
第七届蓝桥杯单片机省赛
第八届蓝桥杯单片机省赛
第九届蓝桥杯单片机省赛
第十届蓝桥杯单片机省赛
第十一届蓝桥杯单片机省赛

  • 16
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值