【51单片机】I2C总线详解 + AT24C02
学习使用的开发板:STC89C52RC/LE52RC
编程软件:Keil5
烧录软件:stc-isp
开发板实图:

文章目录
- AT24C02介绍
- 存储器
- I2C总线介绍
- I2C时序结构
- 数据帧
- AT24C02数据帧
- 编程实例 —— 按键控制数据大小&存储器写入读出
AT24C02介绍
- AT24C02 是一种可以实现掉电不丢失的存储器,可用于保存单片机运行时想永久保存的数据信息
- 存储介质:
E2PROM - 通讯接口:
I2C总线 - 容量:256字节

引脚及应用电路如下:

详细结合I2C通信协议讲解
存储器
存储器大致分为两类:
RAM(Random Access Memory,随机存取存储器),掉电丢失
ROM(Read Only Memory,只读存储器),掉电不丢失

RAM
- SRAM(静态随机存储器):访问时间短,速度快,用于CPU高速缓存(Cache)。
- DRAM(动态随机存储器):访问时间长,速度较慢,用于主存,也就是运行内存。
ROM
- Mask ROM:在制作过程写入,写入后不可更改,类似磁带,只用于读取
- PROM:可改写一次的只读存储器。PROM在出厂时,存储的内容全为1,用户可以根据需要将其中的某些单元写入数据0(部分的PROM相反,出场全为0), 以实现对其“编程”的目的
- EPROM(可擦除可编程ROM):具有可擦除功能,擦除后即可进行再编程,但是缺点是擦除需要使用紫外线照射一定的时间,且时间较长
- E2PROM/E2PROM/EEPROM(电可擦除):可直接用电信号擦除,也可用电信号写入,写入时间为ms级别
- Flash(闪存):属于E2PROM的改进产品。最大特点是必须按块(Block)擦除(每个区块的大小不定,不同厂家的产品有不同的规格), 而EEPROM则可以一次只擦除一个字节(Byte)。
- 硬盘、软盘、光盘:相应控制器 + Flash闪存芯片
I2C总线介绍
单片机常用的通信方式有:UART(串口)、SPI、I2C、DS232
UART常用于两个单片机进行通信,详情参看【51单片机】串口通信
UART有三个缺陷:不能远距离传输、传输速度慢、不能一对多通信。针对这三个缺陷,衍生出了其他的通信方式:
远距离传输:RS232/RS485
传输速度:SPI(加了个clock线提供时钟信号),常用于SD卡,屏幕这类对通信速度要求高的外设
一对多通信:I2C —— 常用于开发板中,单片机与其他外设进行通信
- I2C总线(Inter IC BUS) 是由 Philips 公司开发的一种通用数据总线
- 同步、半双工、带数据应答
- 有两根通信线:SCL(Serial Clock)、SDA(Serial Data)
所有I2C设备的 SCL 连接在一起,SDA 连在一起
设备的 SCL 和 SDA 均要配置成开漏输出模式

开漏输出可参看B站up主工科男孙老师的《推挽 开漏 高阻 这都是谁想出来的词??》
开漏输出可用于适配不同电压的芯片;在总线通信时避免不同设备输出高低电平导致设备烧毁。通过搭配外接上拉电阻输出高低电平
当 SCLKN1OUT 关闭,SCLKIN 为高电平;当 SCLK1OUT 打开,SCLKIN 为低电平
SCL 和 SDA 各添加一个上拉电阻,阻止一般为4.7KΩ左右
上拉电阻可参看B站up主工科男孙老师的《单片机的上拉电阻 到底在拉什么?》
上拉电阻阻值过小,会漏电;阻值过大,驱动能力(将低电平拉高的速度)下降。
当总线挂载设备增多时,可适当降低上拉电阻阻值

开漏输出 和 上拉电阻 的共同作用实现了“线与”的功能,解决多机通信互相干扰的问题
当总线设备较多且通信速度要求高,可以适当降低上拉电阻的阻值
详情参看B站up主工科男孙老师的《单片机I2C通信入门(上):硬件部分有哪些注意点?》
I2C时序结构
因为总线上挂载多个设备,需要约定通信协议才能让通信正确无误,畅通无阻
例如一次读数据通信,需要知道什么时候发起,向谁读数据,对方是否存在,读数据,停止读数据。
I2C通信协议将上述操作都抽象成了一个个时序,最终拼接成数据帧
总共有六个时序结构,如下:
起始条件 —— 数据帧的起始,表示开始通信
- SCL 高电平期间,SDA 从高电平切换到低电平

终止条件 —— 数据帧的结尾,表示通信结束
- SCL 高电平期间,SDA 从低电平切换到高电平

发送一个字节 —— 可发送数据,I2C地址,字地址…
- SCL 低电平期间,主机将数据位依次放到SDA线上(高位在前),然后拉高 SCL,从机(对方设备)将在 SCL 高电平期间读取数据位。SCL 高电平期间 SDA 不允许有数据变化,依次循环上述过程8次,即可发送一个字节数据

注意:先拉低 SCL,再将数据放到 SDA,然后拉高 SCL,此时从机会读取 SDA 数据
接收一个字节 —— 主机(MCU) 接收从机数据
- SCL 低电平期间,从机将数据依次放到 SDA 线上(高位在前),然后拉高 SCL,主机将在 SCL 高电平期间读取数据位,所以SCL高电平期间SDA不允许有数据变化,依次循环上述过程8次,即可接收一个字节(主机在接收之前,需要释放SDA)

注意:先拉高 SDA,释放总线给从机。从机在 SCL 低电平时将数据放到 SDA ,主机拉高 SCL 读取数据,再拉低 SCL 让从机放数据
发送应答 —— 接收从机数据后,告诉从机无误或有误
- 在接收完一个字节之后,主机在下一个时钟发送一位数据、数据0表示应答,数据1表示非应答。发送流程同
发送一位数据

接收应答 —— 主机发送一字节数据后,接收从机的接收应答
- 在主机发送完一个字节之后,主机在下一个时钟接收一位数据,判断从机是否应答,数据0 表示应答,数据1 表示非应答(主机在接收之前,需要释放SDA)

数据帧
I2C数据帧描述具体的一次通信,由上述时序结构拼接而成。
发送一帧数据 —— 向谁发什么

协议格式中第一个字节(为slave address)由7位地址和一位R/W读写位组成的,这字节是个器件地址。
I2C 挂载多个设备,通过从机地址(slave address)标识对哪个设备进行读/写操作
常用I2C接口通用器件的器件地址是由种类型号,及寻址码组成的,共7位。
如格式如下:

-
器件类型:A6 - A3 共4位决定的。这是由半导公司生产时就已固定此类型的了,也就是说这4位已是固定的。本开发板为
1010 -
用户自定义地址码:A2 - A0 共3位。这是由用户自己设置的,通常的作法如EEPROM这些器件是由外部IC的3个引脚所组合电平决定的(用常用的名字如A0,A1,A2)。这也就是寻址码。
AT24C02 的寻址码为 000

-
最低一位就是R/W位,,“0”表示写,“1”表示读(通常读写信号中写上面有一横线,表示低电平)。所以I2C设备通常有两个地址,即读地址和写地址
综上,AT24C02 的写地址为0xA0,读地址为0xA1

回到数据帧,发送一帧数据流程如下
S(起始位) -> 发送从机地址 + W -> 从机返回接收应答 -> 发送一字节数据 -> 从机返回接收应答 … -> P(终止位)
接收一帧数据

S(起始位) -> 发送从机地址 + W -> 从机返回接收应答 -> 接收一字节数据 -> 主机发送应答 -> … -> P(终止位)
复合格式 —— 先发送再接收

AT24C02数据帧
AT24C02是一个存储器,本质就是写入数据和读取数据,其数据帧分为字节写 和 随机读
字节写

存储器内部还有字地址,需要指明将数据写到存储器的哪个位置 —— word address
随机读

需要先告诉 AT24C02 要读哪个地址的数据,再进行读字节数据帧
编程实例 —— 按键控制数据大小&存储器写入读出
按键和数码管部分参看【51单片机】独立按键 和 【51单片机】数码管
首先对I2C的时序结构进行封装,读者可以对照上述时序图,代码如下:
#include <REGX52.h>
#include "Delay.h"sbit I2C_SDA = P2^0;//数据线
sbit I2C_SCL = P2^1;//Clock线/*** @brief I2C开始标志* @parm 无* @retval 无*/
void I2C_Start()
{//起始条件:SCL高电平期间,SDA从高电平切换到低电平//先置SDA = 1,确保从高电平切到低电平I2C_SDA = 1;//SCL置高电平I2C_SCL = 1;//SDA由高电平切换为低电平I2C_SDA = 0;//SCL恢复低电平I2C_SCL = 0;
}
/*** @brief I2C停止标志* @parm 无* @retval 无*/
void I2C_Stop()
{//终止条件:SCL高电平期间,SDA从低电平切换到高电平//先置SDA = 0,确保从低电平切到高电平I2C_SDA = 0;//SCL置高电平I2C_SCL = 1;//SDA由低电平切换为高电平I2C_SDA = 1;
}
/*** @brief I2C发送字节数据* @parm Byte:要发送的数据,范围: 0 ~ 255* @retval 无*/
void I2C_SendByte(unsigned char Byte)
{//发送数据:从高位开始//SCL低电平期间,SDA写数据,SCL拉高后,从机读取数据unsigned char i = 0;for(i = 0; i < 8; ++i){I2C_SDA = Byte & (0x80 >> i);I2C_SCL = 1;//此处要注意若机器频率过快,可能导致从机还没读到数据,SCL就被拉低了I2C_SCL = 0;//需要查看手册。}
}
/*** @brief I2C读取一字节数据* @parm 无* @retval 读取到的数据*/
unsigned char I2C_ReceiveByte()
{//接收数据,从高位开始//SCL高电平时,从SDA读数据unsigned char Byte = 0, i;//将SDA置高电平,释放SDA给从机I2C_SDA = 1;for(i = 0; i < 8; ++i){I2C_SCL = 1;if(I2C_SDA) Byte |= (0x80 >> i);I2C_SCL = 0;}return Byte;
}
/*** @brief 发送应答* @parm AckBit为0表示应答,为1表示非应答* @retval 无*/
void I2C_SendAck(unsigned char AckBit)
{//在读完一个字节后,主机在下一个时钟发送一位数据//数据0表示应答,数据1表示非应答I2C_SDA = AckBit;I2C_SCL = 1;I2C_SCL = 0;
}
/*** @brief 接收应答* @parm 无* @retval Ack为0表示应答,为1表示非应答*/
unsigned char I2C_ReceiveAck()
{//在发完一个字节后,主机在下一个时钟接收一位数据,判断从机是否应答//数据0表示应答,数据1表示非应答(主机在接收之前,需要释放SDA)unsigned char Ack;//释放SDAI2C_SDA = 1;//SCL置高电平I2C_SCL = 1;//读响应Ack = I2C_SDA;//SCL置低电平I2C_SCL = 0;return Ack;
}
然后封装 AT24C02 的两个数据帧,对 I2C 的时序结构进行拼接
#include <REGX52.h>
#include "I2C.h"
//AT24C02的从机地址写,从机地址读需 |= 0x01
#define AT24C02_ADDRESS 0xA0/*** @brief 向AT24C02暂存器写入数据* @parm WordAddress:写到暂存器的什么地址* @parm Data:要写入的数据 * @retval 无*/
void AT24C02_WriteByte(unsigned char WordAddress, unsigned char Data)
{I2C_Start();I2C_SendByte(AT24C02_ADDRESS);I2C_ReceiveAck();I2C_SendByte(WordAddress);I2C_ReceiveAck();I2C_SendByte(Data);I2C_ReceiveAck();I2C_Stop();
}
/*** @brief 从AT24C02读取数据* @parm WordAddress:读什么地址的数据* @retval 读出的数据*/
unsigned char AT24C02_ReadByte(unsigned char WordAddress)
{unsigned char Data;I2C_Start();I2C_SendByte(AT24C02_ADDRESS);I2C_ReceiveAck();I2C_SendByte(WordAddress);I2C_ReceiveAck();I2C_Start();I2C_SendByte(AT24C02_ADDRESS | 0x01);//此处是读I2C_ReceiveAck();Data = I2C_ReceiveByte();I2C_SendAck(1);I2C_Stop();return Data;
}
主程序 —— 监控按键按下,并执行相应操作
- 按键一:变量++
- 按键二:变量–
- 按键三:将变量写入 AT24C02
- 按键四:将数据读出 At24C02
#include <REGX52.h>
#include "AT24C02.h"
#include "SoleKey.h"
#include "LCD1602.h"
#include "Delay.h"void main()
{unsigned char Key = 0;unsigned int Count = 0;LCD_Init();//LCD1602初始化LCD_ShowNum(1, 1, Count, 5);while(1){Key = SoleKey();//获取哪个按键按下if(Key){if(Key == 1)//1按键按下,Count++{Count++;LCD_ShowNum(1, 1, Count, 5);}else if(Key == 2)//2按键按下,Count--{Count--;LCD_ShowNum(1, 1, Count, 5);}else if(Key == 3)//3按键按下,存储数据{AT24C02_WriteByte(0x00, Count / 256);Delayms(5);//因为写速度较慢,不能连续写,会导致数据错误AT24C02_WriteByte(0x01, Count % 256);LCD_ShowNum(1, 1, Count, 5); //显示写入成功LCD_ShowString(2, 1, "Write OK!");Delayms(1000);LCD_ShowString(2, 1, " ");}else if(Key == 4)//4按键按下,读出数据{Count = AT24C02_ReadByte(0x00) * 256 + AT24C02_ReadByte(0x01);LCD_ShowNum(1, 1, Count, 5);//显示读出成功LCD_ShowString(2, 1, "Read OK! ");Delayms(1000);LCD_ShowString(2, 1, " ");}}}
}
效果如下:
【51单片机AT24C02 & I2C通信】
完整项目链接:
AT24C02 存储数据 & I2C通信
以上就是本篇博客的所有内容,感谢你的阅读
如果觉得本篇文章对你有所帮助的话,不妨点个赞支持一下博主,拜托啦,这对我真的很重要。

相关文章:
【51单片机】I2C总线详解 + AT24C02
学习使用的开发板:STC89C52RC/LE52RC 编程软件:Keil5 烧录软件:stc-isp 开发板实图: 文章目录 AT24C02介绍存储器 I2C总线介绍I2C时序结构数据帧AT24C02数据帧 编程实例 —— 按键控制数据大小&存储器写入读出 AT24C02介绍 …...
直接插入排序法
任务描述 本关任务:要求编写程序实现直接插入排序的功能。 相关知识 直接插入排序,是指将无序序列中的各元素依次插入到已经有序的数组中。 假设数组中前 i-1 元素已经有序,现在要将线性表中第 i 个元素插入到前面的有序子表中,…...
mysql中InnoDB索引与MyISAM索引
mysql索引 InnoDB 索引存储 主键索引(聚簇索引) 定义:主键索引是 InnoDB 存储引擎的聚簇索引,它决定了表中数据的物理存储顺序。每个 InnoDB 表都有一个且仅有一个聚簇索引。存储:主键索引的叶子节点直接包含表的数…...
Redis如何保证数据不丢失(可靠性)
本文主要以学习为主,详细参考:微信公众平台 Redis 保证数据不丢失的主要手段有两个: 持久化 多机部署 我们分别来看它们两的具体实现细节。 1.Redis 持久化 持久化是指将数据从内存中存储到持久化存储介质中(如硬盘…...
【计网】物理层学习笔记
【计网】物理层 物理层概述 物理层要实现的功能 在各种传输媒体上传输比特0和1,进而为上面的数据链路层提供透明传输比特流的作用。 物理层接口特性 物理层之下的传输媒体 传输媒体是计网设备之间的物理通路,也称为传输介质。 传输媒体并不包含在…...
vue链接跳转
在 Vue 3 的组合式 API 中,你可以使用 ref 和 setup 函数来实现外部链接跳转功能。 方法 1:使用 click 和 window.open(新标签页跳转) 这种方式在点击时会打开一个新标签页并跳转到外部链接。 <menu-item value"item2&…...
IP地址是电脑自带的吗?是根据什么而决定的
IP地址并非电脑自带,而是由网络运营商或网络服务提供商通过特定的协议和机制进行分配和管理的。要深入理解IP地址的来源和决定因素,我们需要从IP地址的基本概念、分配方式以及影响分配的因素等多个方面进行探讨。 IP地址,即互联网协议地址&am…...
JavaFX史上最全教程 - Shape - JavaFX矩形椭圆
avaFX Shape类定义了常见的形状,如线,矩形,圆,Arc,CubicCurve,Ellipse和QuadCurve。 在场景图上绘制矩形需要宽度,高度和左上角的(x,y)位置。 要在JavaFX中…...
SpringBoot实现的企业资产管理系统
2相关技术 2.1 MYSQL数据库 MySQL是一个真正的多用户、多线程SQL数据库服务器。 是基于SQL的客户/服务器模式的关系数据库管理系统,它的有点有有功能强大、使用简单、管理方便、安全可靠性高、运行速度快、多线程、跨平台性、完全网络化、稳定性等,非常…...
python-读写Excel:openpyxl-(4)下拉选项设置
使用openpyxl库的DataValidation对象方法可添加下拉选择列表。 DataValidation参数说明: type: 数据类型("whole", "decimal", "list", "date", "time", "textLength", "custom"…...
【C++】详解RAII思想与智能指针
🌈 个人主页:谁在夜里看海. 🔥 个人专栏:《C系列》《Linux系列》 ⛰️ 丢掉幻想,准备斗争 目录 引言 内存泄漏 内存泄漏的危害 内存泄漏的处理 一、RAII思想 二、智能指针 1.auto_ptr 实现原理 模拟实现 弊端…...
Qt 环境实现视频和音频播放
在这个示例中,我们将使用 FFmpeg 进行视频和音频的解码,并使用 Qt 的界面进行显示和控制。为了实现音频和视频的解码以及同步显示,我们需要使用 FFmpeg 的解码库进行视频和音频解码,使用 Qt 的 QLabel 显示解码后的视频帧…...
【人工智能训练师】7 大数据处理与应用
大数据处理与应用(Hive技术)(0/100分) 1.本地开发工具连接Hadoop集群 1.本次环境版本为Hadoop2.7.7,对应eclips插件存放于云主机master:/usr/package277/中。 2.本机映射名为hadoop000,云主机Hadoop/Hive的hosts文件中IP需要修改…...
nginx配置文件介绍及示例
一、nginx配置文件一共有main,http,server,location,upstream,stream,events7个块。 step 1: main 块 作用:main 块是 Nginx 配置文件的顶级块,用于设置一些全局的参数和配置&…...
如何在算家云搭建YOLOv5(物体检测)
一、YOLOv5简介 YOLOv5 模型是一种以实时物体检测闻名的计算机视觉模型,由 Ultralytics 开发,并于 2020 年年中发布。它是 YOLO 系列的升级版,继承了 YOLO 系列以实时物体检测能力而著称的特点。 二、模型搭建流程 1.选择模型实例 在应用…...
现场工程师日记-MSYS2迅速部署PostgreSQL主从备份数据库
文章目录 一、概要二、整体架构流程1. 安装 MSYS2 环境2. 安装postgresql 三、技术名词解释1.MSYS22.postgresql 四、技术细节1. 创建主数据库2.添加从数据库复制权限3. 按需修改参数(1)WAL保留空间(2)监听地址 4. 启动主服务器5.…...
使用Element UI实现一个拖拽图片上传,并可以Ctrl + V获取图片实现文件上传
要在 Element UI 的拖拽上传组件中实现 Ctrl V 图片上传功能,可以通过监听键盘事件来捕获粘贴操作,并将粘贴的图片数据上传到服务器。 版本V1,实现获取粘贴板中的文件 注意,本案例需要再你已经安装了Element UI并在项目中正确配…...
私域流量圈层在新消费时代的机遇与挑战:兼论开源 AI 智能名片、2 + 1 链动模式、S2B2C 商城小程序的应用
摘要:本文剖析了私域流量圈层在新消费时代呈现出的独特温度与信任优势,阐述了从传统销售到新消费转型中用户心理的变化。同时,强调了内容对于私域流量的关键作用,并分析开源 AI 智能名片、2 1 链动模式、S2B2C 商城小程序在私域流…...
vxe-vxe-colgroup后端返回数据 对数据进行处理 动态合并分组表头(v-if控制表格渲染(数据请求完成后渲染))
1.html vxe-colgroup循环合并数据;v-if控制表格渲染(数据请求完成后渲染) <template><vxe-table v-if"isTableReady" :data"tableData"><vxe-colgroup title"基本信息"><template v-for…...
ESLint 使用教程(五):从输入 eslint 命令到最终代码被处理,ESLint 中间究竟做了什么工作
前言 ESLint 是现代 JavaScript 开发中不可或缺的代码质量工具。它能够帮助开发者找到并修复代码中的问题,提升代码的可维护性。但是,你可能会好奇:从我们在终端里输入 eslint 命令到最终代码被处理,ESLint 中间究竟做了什么工作…...
使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式
一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明:假设每台服务器已…...
《Qt C++ 与 OpenCV:解锁视频播放程序设计的奥秘》
引言:探索视频播放程序设计之旅 在当今数字化时代,多媒体应用已渗透到我们生活的方方面面,从日常的视频娱乐到专业的视频监控、视频会议系统,视频播放程序作为多媒体应用的核心组成部分,扮演着至关重要的角色。无论是在个人电脑、移动设备还是智能电视等平台上,用户都期望…...
2025年能源电力系统与流体力学国际会议 (EPSFD 2025)
2025年能源电力系统与流体力学国际会议(EPSFD 2025)将于本年度在美丽的杭州盛大召开。作为全球能源、电力系统以及流体力学领域的顶级盛会,EPSFD 2025旨在为来自世界各地的科学家、工程师和研究人员提供一个展示最新研究成果、分享实践经验及…...
质量体系的重要
质量体系是为确保产品、服务或过程质量满足规定要求,由相互关联的要素构成的有机整体。其核心内容可归纳为以下五个方面: 🏛️ 一、组织架构与职责 质量体系明确组织内各部门、岗位的职责与权限,形成层级清晰的管理网络…...
数据链路层的主要功能是什么
数据链路层(OSI模型第2层)的核心功能是在相邻网络节点(如交换机、主机)间提供可靠的数据帧传输服务,主要职责包括: 🔑 核心功能详解: 帧封装与解封装 封装: 将网络层下发…...
【Web 进阶篇】优雅的接口设计:统一响应、全局异常处理与参数校验
系列回顾: 在上一篇中,我们成功地为应用集成了数据库,并使用 Spring Data JPA 实现了基本的 CRUD API。我们的应用现在能“记忆”数据了!但是,如果你仔细审视那些 API,会发现它们还很“粗糙”:有…...
C# 类和继承(抽象类)
抽象类 抽象类是指设计为被继承的类。抽象类只能被用作其他类的基类。 不能创建抽象类的实例。抽象类使用abstract修饰符声明。 抽象类可以包含抽象成员或普通的非抽象成员。抽象类的成员可以是抽象成员和普通带 实现的成员的任意组合。抽象类自己可以派生自另一个抽象类。例…...
Swagger和OpenApi的前世今生
Swagger与OpenAPI的关系演进是API标准化进程中的重要篇章,二者共同塑造了现代RESTful API的开发范式。 本期就扒一扒其技术演进的关键节点与核心逻辑: 🔄 一、起源与初创期:Swagger的诞生(2010-2014) 核心…...
Go 并发编程基础:通道(Channel)的使用
在 Go 中,Channel 是 Goroutine 之间通信的核心机制。它提供了一个线程安全的通信方式,用于在多个 Goroutine 之间传递数据,从而实现高效的并发编程。 本章将介绍 Channel 的基本概念、用法、缓冲、关闭机制以及 select 的使用。 一、Channel…...
LLMs 系列实操科普(1)
写在前面: 本期内容我们继续 Andrej Karpathy 的《How I use LLMs》讲座内容,原视频时长 ~130 分钟,以实操演示主流的一些 LLMs 的使用,由于涉及到实操,实际上并不适合以文字整理,但还是决定尽量整理一份笔…...
