51单片机学习第五课---B站UP主江协科技
1、I2C总线
注意:这里的SCL和SDA都是指总线,即主机来发送开始和停止信号。
2、AT24C02数据存储
main.c
#include <regx52.h>
#include"delay.h"
#include"LCD1602.h"
#include"key.h"
#include"at24c02.h"
unsigned char keynum;
unsigned int num;
void main ()
{
LCD_Init();
LCD_ShowNum(1,1,num,5);
// at24c02_writebyte(1,66) ;//写入一个字地址,子数据,写入的数据掉电不丢失
// Delay(5);//写完不能马上读,at24c02芯片有要求写周期为5ms
// Data=at24c02_readbyte(1);//把读到的子数据赋给Data
// LCD_ShowNum(2,1,Data,3);//再LCD1602上把读到的数据显示出来
while(1)
{
keynum=key();
if(keynum==1){num++;LCD_ShowNum(1,1,num,5);} //按下按键1,num开始加1
if(keynum==2){num--;LCD_ShowNum(1,1,num,5);} //按下按键2,num开始减1
if(keynum==3) //按下按键3,写入num的值,num的值从0~65535
{
at24c02_writebyte(0,num%256);//num是16位,把低八位写入,放在不同的地址
Delay(5);
at24c02_writebyte(1,num/256); //把高八位写入
Delay(5);
LCD_ShowString(2,1,"write ok");
Delay(1000);
LCD_ShowString(2,1," "); //清零
}
if(keynum==4) //按下按键4,读出num的值
{
num=at24c02_readbyte(0);//读取低八位
num=num|(at24c02_readbyte(1)<<8);//先读取高8位,再左移8位,与上低八位???这里at24c02_readbyte(1)是char类型,为8位,左移8位不会溢出吗?
//大整数移位操作都是算术移位(左移的话精度会增加,而不是把移出的位丢掉
LCD_ShowNum(1,1,num,5);
LCD_ShowString(2,1,"read ok");
Delay(1000);
LCD_ShowString(2,1," ");
}
}
}
I2C.C
#include <regx52.h>
sbit I2C_scl=P2^1; //定义端口
sbit I2C_sda=P2^0;
void I2C_start() //开始信号
{
I2C_sda=1;//确保sda是为高电平,虽然一上电sda是为高电平,但是在发送完一个字节且应答后,sda可能为低电平
I2C_scl=1;//这里不能先把scl先拉高,因为如果sda为0,scl在高电平时sda里数据不能变化
I2C_sda=0;
I2C_scl=0; //这里拉低scl主要是开始信号后是发送字节,发送字节前,scl要在低电平下,主机才能把数据放到sda上
}
void I2C_stop()//终止信号
{
I2C_sda=0;
I2C_scl=1;
I2C_sda=1;
}
void I2C_sendbyte(unsigned char byte) //发送字节
{
unsigned char i;
for(i=0;i<8;i++)
{
I2C_sda=byte&(0x80>>i);//发送数据是从最高位开始,单独把最高位与1做与运算,当发送数据是1时还是1
//是0就是0,再右移,直至最低位,8位数据都发送完,即完成一个字节的传输
I2C_scl=1;
I2C_scl=0;//每完成一位数据的传输,scl要拉低,这样主机才能把下一位数据放到sda
}
}
unsigned char I2C_receivebyte()//接收字节
{
unsigned char i,byte=0x00;
I2C_sda=1;//接收字节前,主机要释放sda总线,接下来是从机向主机发送数据
//scl低电平时,从机把数据放到sda,然后scl拉高,主机读取数据
//为什么发送字节不用释放sda总线,因为发送字节时本来sda就是由总线控制,接收字节时,是从机要从
//主机那里拿到控制权,然后把数据放到sda总线上,所以此时主机要释放总线
for(i=0;i<8;i++)
{
I2C_scl=1;//注意这里是拉高scl,因为此前肯定是低的,因为写字节后我们每次都拉低scl,这个期间内从机已经把数据放到sda里了
if(I2C_sda){byte=byte|(0x80>>i);}
I2C_scl=0; //前面做了拉高,后面要拉低,拉低才能从机忘sda中写字节
}
return byte;
}
void I2C_send_ack(unsigned char Ackbit) //当主机发送字节,从机接收字节完后,主机发送应答
{
I2C_sda=Ackbit;
I2C_scl=1; //这里为什么不用先拉低,是因为发送字节后我们每次会拉低,此时scl已经是低的
I2C_scl=0;
}
unsigned char I2C_receive_ack()//当从机发送字节后,主机接收字节完,主机发接收应答
{
unsigned char Ackbit;
I2C_sda=1;
I2C_scl=1;
Ackbit=I2C_sda;
I2C_scl=0;
return Ackbit;
}
at24c02.c
#include<REGX52.h>
#include"I2C.h"
#define at24c02_address 0xA0 //0xA0从机写地址(主机向从机发送数据,从机写入数据),0xA1从机读地址(从机向主句发送数据,主机读数据)
void at24c02_writebyte(unsigned char wordaddress,Data) //写入字节地址,字数据
{
unsigned char ack;
I2C_start();//开始
I2C_sendbyte(at24c02_address);//发送从机写地址
I2C_receive_ack(); //接收应答
I2C_sendbyte(wordaddress); //发送字地址
I2C_receive_ack();
I2C_sendbyte(Data);//发送数据
I2C_receive_ack();
I2C_stop(); //停止
}
unsigned char at24c02_readbyte(unsigned char wordaddress)//读数据(随机读)
{
unsigned char Data;
I2C_start();
I2C_sendbyte(at24c02_address);
I2C_receive_ack();
I2C_sendbyte(wordaddress);
I2C_receive_ack();
I2C_start();
I2C_sendbyte(at24c02_address|0X01);
I2C_receive_ack();
Data=I2C_receivebyte();
I2C_send_ack();
I2C_stop();
return Data;
}
3、秒表(定时器扫描按键数码管)
功能:按下K1秒表开始计时,再按K1,暂停计时,按下K2计时清零,按下K3,AT24C02写入当下秒表的数据,按K4读出AT24C02写入的数据。但我的程序还有问题,就是我的秒写入后,只显示第一次写入的秒,后面再写入,minisec能正常写入和读出,但sec只会显示第一次写入的值,而且再按K3,min会加1,变成1.目前理解不了,后续再修改。
main.c
#include<regx52.h>
#include"Timer0.h"
#include"delay.h"
#include"LCD1602.h"
#include"key.h"
#include"nixie.h"
#include"at24c02.h"
#include"I2C.h"
unsigned char keynum,runflag;
unsigned char min,sec,minisec;
void main ()
{
Timer0_Init();
while(1)
{
keynum=key();
if(keynum==1)
{runflag=!runflag;}
if(keynum==2)
{
min=0;
sec=0;
minisec=0;
}
if(keynum==3)
{
at24c02_writebyte(0,min);
Delay(5);
at24c02_writebyte(1,sec);
Delay(5);
at24c02_writebyte(2,minisec);
Delay(5);
}
if(keynum==4)
{
min=at24c02_readbyte(0);
sec=at24c02_readbyte(1);
minisec=at24c02_readbyte(2);
}
nixie_setbuff(0,min/10);
nixie_setbuff(1,min%10);
nixie_setbuff(2,11);
nixie_setbuff(3,sec/10);
nixie_setbuff(4,sec%10);
nixie_setbuff(5,11);
nixie_setbuff(6,minisec/10);
nixie_setbuff(7,minisec%10);
}
}
void sec_loop()
{
if(runflag)
{ minisec++;
if (minisec>=100){minisec=0;sec++;}
if (sec>=60){sec=0;min++;}
if (min>=60){min=0;}
}
}
void Timer0_Routine() interrupt 1
{
static unsigned int count1,count2,count3; //
TL0 = 0x18;
TH0 = 0xFC;
count1++;
if(count1>=20)
{
count1=0;
key_loop();}
count2++;
if(count2>=2)
{
count2=0;
nixie_loop();
}
count3++;
if(count3>=10)
{
count3=0;
sec_loop();
}
}
nixie.c
#include <regx52.h>
#include"delay.h"
unsigned char nixie_buff[9]={0,10,10,10,10,10,10,10};
unsigned char nixieTable[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x00,0x40};
void nixie_setbuff(unsigned char location,number)
{
nixie_buff[location]= number;
}
void nixie (unsigned char location,number)
{
P0=0x00;
switch(location)
{
case 0:P2_2=1;P2_3=1;P2_4=1;break;
case 1:P2_2=0;P2_3=1;P2_4=1;break;
case 2:P2_2=1;P2_3=0;P2_4=1;break;
case 3:P2_2=0;P2_3=0;P2_4=1;break;
case 4:P2_2=1;P2_3=1;P2_4=0;break;
case 5:P2_2=0;P2_3=1;P2_4=0;break;
case 6:P2_2=1;P2_3=0;P2_4=0;break;
case 7:P2_2=0;P2_3=0;P2_4=0;break;
}
P0=nixieTable[number];
}
void nixie_loop()
{ static char i;
nixie (i,nixie_buff[i]);
i++;
if(i>7){i=0;}
}
注明出处链接:I2C讲解https://handsome-man.blog.csdn.net/article/details/123673285?fromshare=blogdetail&sharetype=blogdetail&sharerId=123673285&sharerefer=PC&sharesource=m0_51664996&sharefrom=from_link