风筝
发表于: 2018-12-25 21:54:19 | 显示全部楼层

在之前的文章中,我们介绍了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通信的优点是可以将多个从机连接到一个主机。

I2C-Communication-Working.png


完整的通信通过这两条导线进行,即串行时钟(SCL)和串行数据(SDA)。

●    串行时钟(SCL):与主设备共享主设备生成的时钟信号

●    串行数据(SDA):在主机和从机之间发送数据。


在任何给定时间,只有主机才能启动通信。由于总线中有多个从站,因此主站必须使用不同的地址来引用每个从站。当被寻址时,只有具有该特定地址的从机将应答该信息,而其他地址继续退出。这样我们就可以使用相同的总线与多个设备进行通信。


I2C的电压电平未预定义。 I2C通信灵活,意味着由5v电源供电的器件,可以使用5v用于I2C,3.3v器件可以使用3v进行I2C通信。但是,如果两个运行在不同电压下的设备需要使用I2C进行通信呢? 5V I2C总线不能与3.3V器件连接。在这种情况下,电压移位器用于匹配两个I2C总线之间的电压电平。


有一些条件可以构成传输。传输的初始化从SDA的下降沿开始,在下图中定义为“START”条件,其中主机将SCL设为高电平,同时将SDA设置为低电平。如下图所示,

I2C-Start-Stop-Condition.png


SDA的下降沿是START条件的硬件触发。在此之后,同一总线上的所有设备都进入监听模式。


同样的,SDA的上升沿停止传输,在上图中显示为“STOP”条件,其中主机将SCL置为高电平并且还释放SDA以变为高电平。因此,SDA的上升沿会阻止传输。

I2C-First-Byte.png

R / W位表示后续字节的传输方向,如果为高电平表示从机将发送,如果为低则表示主机将发送。


每个位在每个时钟周期发送,因此传输一个字节需要8个时钟周期。在发送或接收每个字节之后,保持第九个时钟周期用于ACK / NACK(确认/未确认)。该ACK位由从机或主机根据情况生成。对于ACK位,SDA在第9个时钟周期由主机或从机设置为低电平。所以它被认为是低,否则NACK。

Introduction-to-I2C-Message.png


在哪里使用I2C通信?

I2C通信仅用于短距离通信。它在某种程度上肯定是可靠的,因为它具有同步的时钟脉冲以使其智能化。该协议主要用于与必须向主设备发送信息的传感器或其他设备进行通信。当微控制器必须使用最少的导线与许多其他从模块通信时非常方便。如果您正在寻找远程通信,您应该尝试RS232,如果您正在寻找更可靠的通信,您应该尝试SPI协议。


Arduino中的I2C

下图显示了Arduino UNO中的I2C引脚。

I2C-in-Arduino.png

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电位器

●    面包板

●    连接导线


电路原理图

Circuit-Diagram-for-I2C-Communication-in-Arduino.png


工作过程

这里为了演示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的值将发生变化,反之亦然。

I2C-Communication-in-Arduino.jpg

跳转到指定楼层
回复

使用道具 举报

风筝
发表于: 2018-12-25 22:25:27 | 显示全部楼层

Arduino中的I2C编程

本篇文章有两个程序,一个用于主机Arduino,另一个用于从机Arduino。在文章的结尾处提供了整个的文件代码。


主机Arduino编程介绍

1.首先,我们需要包含用于使用I2C通信功能的Wire库和用于使用LCD功能的LCD库。还需要为1602 LCD定义LCD引脚。

  1. #include<Wire.h>   
  2. #include<LiquidCrystal.h>
  3. LiquidCrystal lcd(2, 7, 8, 9, 10, 11);
复制代码

2.在void setup()函数中,

我们以波特率9600启动串行通信。

  1. Serial.begin(9600);               
复制代码

接下来在引脚(A4,A5)上启动I2C通信

  1. Wire.begin();       //Begins I2C communication at pin (A4,A5)
复制代码

接下来我们在1602模式下初始化LCD显示模块并显示欢迎信息,然后在五秒后清除。

  1. lcd.begin(16,2);                           //Initilize LCD display
  2.   lcd.setCursor(0,0);                        //Sets Cursor at first line of Display
  3.   lcd.print("Circuit Digest");               //Prints CIRCUIT DIGEST in LCD
  4.   lcd.setCursor(0,1);                        //Sets Cursor at second line of Display
  5.   lcd.print("I2C 2 ARDUINO");                //Prints I2C ARDUINO in LCD
  6.   delay(5000);                               //Delay for 5 seconds
  7.   lcd.clear();                               //Clears LCD display
复制代码

3.在void loop()函数中

首先,我们需要从Slave获取数据,因此我们使用requestFrom()和从地址8,我们请求一个字节

  1. Wire.requestFrom(8,1);                          
复制代码

使用Wire.read()读取接收的值

  1. byte MasterReceive = Wire.read();      
复制代码

接下来,我们需要读取连接到引脚A0的主机arduino电位器的模拟值

  1. int potvalue = analogRead(A0);
复制代码

我们将该值转换为0到127的字节。

  1. byte MasterSend = map(potvalue,0,1023,0,127);         
复制代码

接下来我们需要发送转换后的值,使用8地址开始从机sarduino的传输

  1. Wire.beginTransmission(8);                          
  2. Wire.write(MasterSend);                       
  3. Wire.endTransmission();     
复制代码

接下来,我们显示来自从机arduino的接收值,延迟为500微秒,我们不断接收并显示这些值。

  1. lcd.setCursor(0,0);            //Sets Currsor at line one of LCD
  2. lcd.print(">>  Master  <<");      //Prints >> Master << at LCD
  3. lcd.setCursor(0,1);                      //Sets Cursor at line two of LCD
  4. lcd.print("SlaveVal:");                //Prints SlaveVal: in LCD
  5. lcd.print(MasterReceive);  //Prints MasterReceive in LCD received from Slave
  6. Serial.println("Master Received From Slave");    //Prints in Serial Monitor
  7. Serial.println(MasterReceive);
  8. delay(500);                                    
  9. lcd.clear();
复制代码

从机Arduino编程介绍

1.与主机设备相同,首先我们需要包含用于使用I2C通信功能的Wire库和用于使用LCD功能的LCD库。还为1602 LCD定义LCD引脚。

  1. #include<Wire.h>   
  2. #include<LiquidCrystal.h>      
  3. LiquidCrystal lcd(2, 7, 8, 9, 10, 11);
复制代码

2.  在void setup()函数中,

我们以波特率9600启动串行通信。

  1. Serial.begin(9600);
复制代码

接下来在引脚(A4,A5)上启动I2C通信,从地址设定为8。这里指定从地址非常重要。

  1. Wire.begin(8);
复制代码

接下来,当从机从主机接收值并且主机请求从机的值时,我们需要调用该函数

  1. Wire.onReceive(receiveEvent);         
  2. Wire.onRequest(requestEvent);  
复制代码

接下来我们在16X2模式下初始化LCD显示模块并显示欢迎信息,然后在五秒后清除。

  1. lcd.begin(16,2);                           //Initilize LCD display
  2.   lcd.setCursor(0,0);                        //Sets Cursor at first line of Display
  3.   lcd.print("Circuit Digest");               //Prints CIRCUIT DIGEST in LCD
  4.   lcd.setCursor(0,1);                        //Sets Cursor at second line of Display
  5.   lcd.print("I2C 2 ARDUINO");                //Prints I2C ARDUINO in LCD
  6.   delay(5000);                               //Delay for 5 seconds
  7.   lcd.clear();                               //Clears LCD display
复制代码

3.接下来,我们有两个函数,一个用于请求事件,另一个用于接收事件


对于请求事件

当主机从从机请求值时,将执行该函数。此函数从从机电位器获取输入值并以7位转换,然后将该值发送给主机。

  1. void requestEvent()  
  2. {
  3.   int potvalue = analogRead(A0);     
  4.   byte SlaveSend = map(potvalue,0,1023,0,127);   
  5.   Wire.write(SlaveSend);        
  6. }
复制代码

对于接收事件

当主机通过从机地址(8)向从机发送数据时,将执行该函数。此函数从主机读取接收的值并存储在byte类型的变量中。

  1. void receiveEvent (int howMany)
  2. {
  3. SlaveReceived = Wire.read();                  
  4. }
复制代码

4.  在void loop()函数中:

我们在LCD显示模块中连续显示主设备的接收值。

  1. void loop(void)
  2. {
  3.   lcd.setCursor(0,0);                              //Sets Currsor at line one of LCD
  4.   lcd.print(">>  Slave  <<");                      //Prints >> Slave << at LCD
  5.   lcd.setCursor(0,1);                              //Sets Cursor at line two of LCD
  6.   lcd.print("MasterVal:");                         //Prints MasterVal: in LCD
  7.   lcd.print(SlaveReceived);                        //Prints SlaveReceived value in LCD received from Master
  8.   Serial.println("Slave Received From Master:");   //Prints in Serial Monitor
  9.   Serial.println(SlaveReceived);
  10.   delay(500);
  11.   lcd.clear();
  12. }
复制代码

通过旋转一侧的电位器,您可以在另一侧看到LCD上的变化值:

Testing-I2C-Communication-in-Arduino.jpg


以上就是在Arduino中进行I2C通信的方式,这里我们不仅使用两个Arduino开发板来演示通过I2C通信发送数据,而且还演示了接收数据。所以现在你可以将任何I2C传感器连接到Arduino。


代码

以下是本篇文章的完整代码: main.rar (1.29 KB, 下载次数: 140)

回复

使用道具 举报

KEFENG1001
发表于: 2020-8-17 23:06:17 | 显示全部楼层

先下载下来,好好学习一下,感谢分享。
回复

使用道具 举报

碌碌无为
发表于: 2020-10-20 21:37:47 | 显示全部楼层

感谢楼主分享,学习一下。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

主题 716 | 回复: 1503



手机版|

GMT+8, 2025-1-8 23:47 , Processed in 0.039105 second(s), 6 queries , Gzip On, MemCache On. Powered by Discuz! X3.5

YiBoard一板网 © 2015-2022 地址:河北省石家庄市长安区高营大街 ( 冀ICP备18020117号 )

快速回复 返回顶部 返回列表