当前位置: 首页 > article >正文

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


http://www.kler.cn/a/328303.html

相关文章:

  • git合并分支
  • 微众银行申请专利:不过分丢失泛用能力,提高语音大模型对困难样本学习效率
  • C++ —— string类(上)
  • Java 反射机制详解
  • 任务管理功能拆解——如何高效管理项目任务?
  • 15-大模型 RAG 经验篇
  • 【网络安全】消息鉴别
  • 五.海量数据实时分析-FlinkCDC+DorisConnector实现数据的全量增量同步
  • Require:基于雪花算法完成一个局部随机,全局离散没有热点切唯一的数值Id生成器。
  • FileLink跨网文件交换:高效、安全、灵活的企业文件传输新方案
  • 力扣10.1
  • 5QI(5G QoS Identifier)
  • 《Linux从小白到高手》理论篇(二):Linux的目录结构和磁盘管理
  • 基于贝叶斯优化CNN-GRU网络的数据分类识别算法matlab仿真
  • python画图|自制渐变柱状图
  • 鸿蒙开发(NEXT/API 12)【穿戴设备信息查询】手机侧应用开发
  • 影院管理新篇章:小徐的Spring Boot应用
  • 低代码时代的企业信息化:规范与标准化的重要性
  • Redis: Sentinel哨兵监控架构及环境搭建
  • 通信工程学习:什么是LAN局域网、MAN城域网、WAN广域网
  • HarmonyOS Next应用开发——@build构建函数的使用
  • 每天一个数据分析题(四百九十一)- 主成分分析与因子分析
  • linux下recoketmq安装教程
  • JVM有哪些参数以及如何使用
  • 基于Java+SQL Server2008开发的(CS界面)个人财物管理系统
  • 【深度学习】(6)--图像数据增强