在之前的文章中,我们介绍了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中的引脚
SDAA4
SCLA5
在开始使用两个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的值将发生变化,反之亦然。
Arduino中的I2C编程
本篇文章有两个程序,一个用于主机Arduino,另一个用于从机Arduino。
主机Arduino编程介绍
1.首先,我们需要包含用于使用I2C通信功能的Wire库和用于使用LCD功能的LCD库。还需要为1602 LCD定义LCD引脚。
#include《Wire.h》
#include《LiquidCrystal.h》
LiquidCrystal lcd(2, 7, 8, 9, 10, 11);
2.在void setup()函数中,
我们以波特率9600启动串行通信。
Serial.begin(9600);
接下来在引脚(A4,A5)上启动I2C通信
Wire.begin(); //Begins I2C communication at pin (A4,A5)
接下来我们在1602模式下初始化LCD显示模块并显示欢迎信息,然后在五秒后清除。
lcd.begin(16,2); //Initilize LCD display
lcd.setCursor(0,0); //Sets Cursor at first line of Display
lcd.print(“Circuit Digest”); //Prints CIRCUIT DIGEST in LCD
lcd.setCursor(0,1); //Sets Cursor at second line of Display
lcd.print(“I2C 2 ARDUINO”); //Prints I2C ARDUINO in LCD
delay(5000); //Delay for 5 seconds
lcd.clear(); //Clears LCD display
3.在void loop()函数中
首先,我们需要从Slave获取数据,因此我们使用requestFrom()和从地址8,我们请求一个字节
Wire.requestFrom(8,1);
使用Wire.read()读取接收的值
byte MasterReceive = Wire.read();
接下来,我们需要读取连接到引脚A0的主机arduino电位器的模拟值
int potvalue = analogRead(A0);
我们将该值转换为0到127的字节。
byte MasterSend = map(potvalue,0,1023,0,127);
接下来我们需要发送转换后的值,使用8地址开始从机sarduino的传输
Wire.beginTransmission(8);
Wire.write(MasterSend);
Wire.endTransmission();
接下来,我们显示来自从机arduino的接收值,延迟为500微秒,我们不断接收并显示这些值。
lcd.setCursor(0,0); //Sets Currsor at line one of LCD
lcd.print(“》》 Master 《《”); //Prints 》》 Master 《《 at LCD
lcd.setCursor(0,1); //Sets Cursor at line two of LCD
lcd.print(“SlaveVal:”); //Prints SlaveVal: in LCD
lcd.print(MasterReceive); //Prints MasterReceive in LCD received from Slave
Serial.println(“Master Received From Slave”); //Prints in Serial Monitor
Serial.println(MasterReceive);
delay(500);
lcd.clear();
从机Arduino编程介绍
1.与主机设备相同,首先我们需要包含用于使用I2C通信功能的Wire库和用于使用LCD功能的LCD库。还为1602 LCD定义LCD引脚。
#include《Wire.h》
#include《LiquidCrystal.h》
LiquidCrystal lcd(2, 7, 8, 9, 10, 11);
2. 在void setup()函数中,
我们以波特率9600启动串行通信。
Serial.begin(9600);
接下来在引脚(A4,A5)上启动I2C通信,从地址设定为8。这里指定从地址非常重要。
Wire.begin(8);
接下来,当从机从主机接收值并且主机请求从机的值时,我们需要调用该函数
Wire.onReceive(receiveEvent);
Wire.onRequest(requestEvent);
接下来我们在16X2模式下初始化LCD显示模块并显示欢迎信息,然后在五秒后清除。
lcd.begin(16,2); //Initilize LCD display
lcd.setCursor(0,0); //Sets Cursor at first line of Display
lcd.print(“Circuit Digest”); //Prints CIRCUIT DIGEST in LCD
lcd.setCursor(0,1); //Sets Cursor at second line of Display
lcd.print(“I2C 2 ARDUINO”); //Prints I2C ARDUINO in LCD
delay(5000); //Delay for 5 seconds
lcd.clear(); //Clears LCD display
3.接下来,我们有两个函数,一个用于请求事件,另一个用于接收事件
对于请求事件
当主机从从机请求值时,将执行该函数。此函数从从机电位器获取输入值并以7位转换,然后将该值发送给主机。
void requestEvent()
{
int potvalue = analogRead(A0);
byte SlaveSend = map(potvalue,0,1023,0,127);
Wire.write(SlaveSend);
}
对于接收事件
当主机通过从机地址(8)向从机发送数据时,将执行该函数。此函数从主机读取接收的值并存储在byte类型的变量中。
void receiveEvent (int howMany)
{
SlaveReceived = Wire.read();
}
4. 在void loop()函数中:
我们在LCD显示模块中连续显示主设备的接收值。
void loop(void)
{
lcd.setCursor(0,0); //Sets Currsor at line one of LCD
lcd.print(“》》 Slave 《《”); //Prints 》》 Slave 《《 at LCD
lcd.setCursor(0,1); //Sets Cursor at line two of LCD
lcd.print(“MasterVal:”); //Prints MasterVal: in LCD
lcd.print(SlaveReceived); //Prints SlaveReceived value in LCD received from Master
Serial.println(“Slave Received From Master:”); //Prints in Serial Monitor
Serial.println(SlaveReceived);
delay(500);
lcd.clear();
}
通过旋转一侧的电位器,您可以在另一侧看到LCD上的变化值:
以上就是在Arduino中进行I2C通信的方式,这里我们不仅使用两个Arduino开发板来演示通过I2C通信发送数据,而且还演示了接收数据。所以现在你可以将任何I2C传感器连接到Arduino。
编辑:hfy
-
lcd
+关注
关注
34文章
4407浏览量
166964 -
电位器
+关注
关注
14文章
1005浏览量
66688 -
I2C
+关注
关注
28文章
1477浏览量
123125 -
Arduino
+关注
关注
187文章
6461浏览量
186549
发布评论请先 登录
相关推荐
评论