|
在之前的文章中,我们介绍了Arduino之间的SPI通信。今天我们将学习另一种串行通信协议:I2C(内部集成电路)。比较I2C和SPI,I2C只有两条线,而SPI使用四条,I2C可以有多个主机和从机,而SPI只能有一个主机和多个从机。因此,如果项目中有多个微控制器需要作为主机,那么就采用I2C。 I2C通信通常用于与陀螺仪、加速度计、气压传感器、LED显示器等进行通信。
在本篇文章中,我们将使用I2C总线在两个arduino开发板之间进行通信,并且使用电位计将值(0到127)相互发送。这些值将显示在连接到每个Arduino的1602液晶显示屏上。文章中,一个Arduino开发板作为主机,另一个开发板作为从机。现在让我们从关于I2C通信的介绍开始吧。
什么是I2C通信协议? 术语IIC代表“Inter Integrated Circuits”。它通常表示为I2C或IIC,甚至在某些地方表示为2线接口协议(TWI),但它们代表的含义是一样的。 I2C是同步通信协议,也就是说共享信息的设备必须共享公共时钟信号。它只有两根线来共享信息,其中一根用于时钟信号,另一根用于发送和接收数据。
I2C通信如何工作? I2C通信最初由Phillips引入。如前所述,它有两根导线,这两根导线将连接在两个设备上。这里一个设备称为主机,另一个设备称为从机。通信应该并且将始终发生在一个主机和一个从机之间。 I2C通信的优点是可以将多个从机连接到一个主机。
完整的通信通过这两条导线进行,即串行时钟(SCL)和串行数据(SDA)。 ● 串行时钟(SCL):与主设备共享主设备生成的时钟信号 ● 串行数据(SDA):在主机和从机之间发送数据。
在任何给定时间,只有主机才能启动通信。由于总线中有多个从站,因此主站必须使用不同的地址来引用每个从站。当被寻址时,只有具有该特定地址的从机将应答该信息,而其他地址继续退出。这样我们就可以使用相同的总线与多个设备进行通信。
I2C的电压电平未预定义。 I2C通信灵活,意味着由5v电源供电的器件,可以使用5v用于I2C,3.3v器件可以使用3v进行I2C通信。但是,如果两个运行在不同电压下的设备需要使用I2C进行通信呢? 5V I2C总线不能与3.3V器件连接。在这种情况下,电压移位器用于匹配两个I2C总线之间的电压电平。
有一些条件可以构成传输。传输的初始化从SDA的下降沿开始,在下图中定义为“START”条件,其中主机将SCL设为高电平,同时将SDA设置为低电平。如下图所示,
SDA的下降沿是START条件的硬件触发。在此之后,同一总线上的所有设备都进入监听模式。
同样的,SDA的上升沿停止传输,在上图中显示为“STOP”条件,其中主机将SCL置为高电平并且还释放SDA以变为高电平。因此,SDA的上升沿会阻止传输。
R / W位表示后续字节的传输方向,如果为高电平表示从机将发送,如果为低则表示主机将发送。
每个位在每个时钟周期发送,因此传输一个字节需要8个时钟周期。在发送或接收每个字节之后,保持第九个时钟周期用于ACK / NACK(确认/未确认)。该ACK位由从机或主机根据情况生成。对于ACK位,SDA在第9个时钟周期由主机或从机设置为低电平。所以它被认为是低,否则NACK。
在哪里使用I2C通信? I2C通信仅用于短距离通信。它在某种程度上肯定是可靠的,因为它具有同步的时钟脉冲以使其智能化。该协议主要用于与必须向主设备发送信息的传感器或其他设备进行通信。当微控制器必须使用最少的导线与许多其他从模块通信时非常方便。如果您正在寻找远程通信,您应该尝试RS232,如果您正在寻找更可靠的通信,您应该尝试SPI协议。
Arduino中的I2C 下图显示了Arduino UNO中的I2C引脚。
I2C总线 | Arduino中的引脚 | SDA | A4 | SCL | A5 |
在开始使用两个Arduino编程I2C之前,我们需要了解Arduino IDE中使用的Wire库。
库<Wire.h>包含在程序中,用于使用以下I2C通信函数。 1. Wire.begin(address): 用途:该库用于与I2C设备进行通信。初始化Wire库,并作为从机或主机加入I2C总线。 address:7位从机地址是可选的,如果未指定地址,类似[Wire.begin()],将作为主机加入总线。
2. Wire.read(): 用途:该函数用于读取从主机或从机接收的字节,该字节是在调用requestFrom()后从一个从机发送到主设备的字节,或从主设备发送到从机的字节。
3. Wire.write(): 用途:该函数用于将数据写入从机或主机。 从机到主机:当主站中使用Wire.RequestFrom()时,从机将数据写入主机。 主机到从机:从主机到从机的传输,Wire.write()用在调用Wire.beginTransmission()和Wire.endTransmission()之间。 Wire.write()可以写成: ◾ Wire.write(value) value:要作为单个字节发送的值。 ◾ Wire.write(string): string:要作为一系列字节发送的字符串。 ◾ Wire.write(data,length): data:要作为字节发送的数据数组 length:要传输的字节数。
4. Wire.beginTransmission(address): 用途:该函数用于开始使用给定的从地址传输到I2C设备。随后,使用write()函数构建用于传输的字节队列,然后通过调用endTransmission()函数传输它们。 address:发送设备的7位地址。
5. Wire.endTransmission(); 用途:此函数用于结束由beginTransmission()发起的从机的传输,并传输由Wire.write()排队的字节。
6. Wire.onRequest(); 用途:当主设备使用Wire.requestFrom()请求来自从设备的数据时,将调用此函数。这里我们可以包含Wire.write()函数来向主机发送数据。
7. Wire.onReceive(); 用途:当从设备从主设备接收数据时,将调用此函数。这里我们可以包含Wire.read();用于读取从主站发送的数据的函数。
8. Wire.requestFrom(addres,quantity); 用途:该函数在主设备中用于从从设备请求字节。函数Wire.read()用于读取从设备发送的数据。 address:要从中请求字节的设备的7位地址 quantity:要请求的字节数
需要的组件 ● Arduino Uno开发板 ● 1602 LCD显示模块 ● 10K电位器 ● 面包板 ● 连接导线
电路原理图
工作过程 这里为了演示Arduino中的I2C通信,我们使用两个Arduino UNO和两个1602 LCD显示器相互连接,并在两个arduino开发板上使用两个电位器来确定从主设备到从设备和从设备到主设备的发送值(0到127)通过改变电位器。
我们使用电位器将arduino引脚A0的输入模拟值从(0到5V)转换为模拟到数字值(0到1023)。然后,这些ADC值进一步转换为(0到127),因为我们只能通过I2C通信发送7位数据。 I2C通信通过arduino的A4和A5引脚上的两条线进行。
通过改变主机的电位器,从机Arduino开发板的LCD的值将发生变化,反之亦然。
|