古人学问无遗力,少壮工夫老始成。这篇文章主要讲述一起玩转树莓派(11)——使用LCD屏相关的知识,希望能为你提供帮助。
一起玩转树莓派(11)——使用LCD屏通过本系列博客前几篇文章的介绍,我们已经体验过了许多传感器元件,它们大多非常简单,可以直接对其进行数据读取或写入,无需复杂的指令配置。本篇博客,我们将介绍一个相对复杂的元件:LCD屏。当今大多数常见的电子设备为了便于用户操作,都会配备一块LCD液晶显示屏,用户通过屏幕可以获取到设备的相关信息方便使用。下面,我们将尝试使用树莓派来在LCD屏上展示信息。
一、LCD 1602LCD屏是Liquid Crystal Display的简称,即液晶显示屏。LCD 1602是一种点阵式的给付型液晶显示屏,其型号为1602本身也是有意义的,表示其可以显示2行信息,每行可以显示16个字符。LCD1602最多可以显示32个字符,价格上也并不昂贵,十几元钱即可买到。LCD1602有16个引脚,如下图所示:
文章图片
LCD1602的16个引脚看上去很多,但实际上使用起来并不复杂,我们首先将上图这些引脚的功能来介绍一下。
- 引脚1:接地引脚
- 引脚2:接5V电压
- 引脚3:VE引脚为屏幕对比度调整引脚,接地时对比度最大,接正极电源时对比度最小。
- 引脚4:RS引脚为功能模式引脚(也被称为寄存器选择引脚),为其加高电平时为数据模式(存取屏幕展示的数据),为其加低电平时为指令模式(读取指令)。
- 引脚5:RW引脚为读写模式引脚,为其加高电平时为读操作,为其加低电平时为写操作。
- 引脚6:Enable引脚,此引脚用来触发动作,负跳变时进行数据处理或指令的执行。
- 引脚7-引脚14:这8个引脚为数据引脚,用来进行数据传输。
- 引脚15:背光电源引脚。
- 引脚16:背光接地引脚。
现在请你务必将上面所介绍的内容完全理解,否则后面的内容可能会更加令你迷惑。对于LCD1206的读数据模式和写数据模式你应该没有什么疑惑,只要通过引脚4和引脚5设置正确的模式后,再通过GPIO来写和读引脚7到引脚14的电平数据,即可得到一个8位的数据。我们核心需要理解的是指令模式,LCD1602的指令集如下:
文章图片
上图中的RS和R/W就是引脚4和引脚5,其控制模式,与指令本身无关,我们可以先不关心。从DB7到DB0是真正的指令部分。我们下面来逐一介绍。
1. 指令一:0000 0001
清屏指令,响应时间为1.53ms。
2.指令二:0000 001- (最后一位’-‘表示0和1都可以,不被关心)
光标归位指令,执行后光标的位置会回到起点,但是数据寄存器中的数据不会清空。
3.指令三:0000 01[I/D][SH]
光标移动模式设置指令,I/D和SH两个控制为光标或屏幕移动模式。
I/D设置为0:每次读取一个字符后光标左移。
I/D设置为1:每次读取一个字符后光标右移。
SH设置为0:屏幕不移动。
SH设置为1:屏幕移动,方向与I/D的设置一致。
4.指令四:0000 1[D][C][B]
显示模式设置指令,D,C,B这三个位分别设置主显示功能,光标显示功能,光标闪烁功能。
D:设置为0则关闭屏幕,设置为1开启屏幕。
C:设置为0关闭光标,设置为1显示光标。
B:设置为0光标不闪烁,设置为1光标闪烁。
【一起玩转树莓派(11)——使用LCD屏】5.指令五:0001[S/C][R/L]--
设置光标和显示屏移动方向。
S/C设置为0时,R/L设置为0则光标左移,RL设置为1时光标右移。
S/C设置为1时,R/L设置为0则显示内容左移,R/L设置为1则显示内容右移。
6.指令六:001[DL] [N][F]--
功能模式设置指令。
DL:设置为1时采用8位总线读数据,设置为0时采用4位总线读数据。
N:设置为0时是单行显示模式,设置为1时是双行显示模式。
F:设置为0时为5*8的点阵字符,设置为1时为5*11的点阵字符。
7.指令七:01[A5][A4] [A3][A2][A1][A0]
设置下一个字符要显示的位置。A5位设置要定位到的行,A0到A4位定位要显示的位置,取值为0-16之间。
8.指令八:1[A6][A5][A4] [A3][A2][A1][A0]
数据寄存器地址设置。
了解了LD1602显示屏上面的指令用法,我们就可以编程来控制显示屏显示的内容了。
二、带I2C模块的LCD 1602前面我们说过,LCD1602有16个引脚。原则上我们已经可以使用树莓派来控制显示屏的显示了,但是16个引脚全部连接到树莓派会使接线十分的复杂,而且程序代码的编写也非常繁琐,要对太多的GPIO引脚进行操作,十分不便。本次实验,我们采用的是带I2C模块的LCD元件,I2C模块本身将一些独立的功能进行了封装,通过I2C模块,我们可以以8位数据为标准来传输任何我们想要执行的指令或让LCD显示的字符,非常方便。
文章图片
如上图所示,有了IC2模块的LCD1602元件,只需要4个引脚即可实现显示功能。下面我们来分析下如何使用此I2C模块。
首先关于I2C通信的相关内容,之前博客已经有详细的介绍,这里不再赘述。我们先来介绍下为何通过4个引脚通过I2C总线传输8位的数据集合实现所有功能。
LCD引脚1与引脚2:用I2C模块的电源引脚和接地引脚代替。
LCD引脚15和引脚16:LCD的这两个引脚功能为控制背光,此逻辑被封装进了I2C模块中,I2C模块每次写入的8位数据中的第4位用来控制背光。
LCD的引脚3:LCD的此引脚用来设置显示的对比度,在I2C模块中,通过一个可调节的电阻来实现此功能,在后面的实验中如果发现屏幕显示不清,可以尝试调节此电阻器。
LCD的引脚4,引脚5和引脚6:这几个功能引脚也被封装进了I2C模块中,I2C模块每次写入的8位数据中的第1位,的2位和第3位分别用来控制这些引脚。
LCD剩下的数据引脚的数据由I2C传输的8位数据中的高4位来对应,在LCD的8位数据模式下,I2C分两次传输一次完整的数据,前传输的4位为LCD所需数据的低4位,后传输的数据为LCD所需数据的高4位。在LCD的4位数据模式下,因为LCD需要获取到完整的8位数据,因此也需要通过两次数据传输,只是此时先传输的数据为LCD所需数据的高4位,后传输的数据为LCD所需数据的低4位,这点需要特别注意。
下面总结了I2C在传输数据时每一位的意义:
第8位 | 第7位 | 第6位 | 第5位 | 第4位 | 第3位 | 第2位 | 第1位 |
---|---|---|---|---|---|---|---|
数据/指令 | 数据/指令 | 数据/指令 | 数据/指令 | 背光控制位 | Enable控制位 | RW控制位 | RS控制位 |
LCD | 树莓派 |
---|---|
GND | GND |
VCC | +5V |
SDA | 树莓派SDA功能引脚 |
SCL | 树莓派SCL功能引脚 |
sudo i2cdetect -y 1
输出入下图所示:
文章图片
可以看到,目前我们只连接了一个I2C设备,设备号为27。
由于背光的控制位相对独立,我们封装单独的函数来处理,如下:
# 是否开启背光 由PCF8574T的低4位中的第4位决定
BLEN = 1
# 补充背光控制位
def addBlenControl(data):
global BLEN
tmpData = https://www.songbingjia.com/android/data
if BLEN:
# 将第4位背光控制位强制设置1
tmpData = data | 0b00001000
else:
# 将第4位背光控制位强制设置为0
tmpData = data &
0b11110111
return tmpData
同理,可以将Enable位的控制,RS位的控制都封装成函数:
# 补充Enable控制位
def addEnableControl(data, high):
tempData = https://www.songbingjia.com/android/data
# 第3位控制Enable
if high:
tempData |= 0b00000100
else:
tempData &
= 0b11111011
return tempData# 补充RS控制位
def addRSControl(data, high):
tempData = data
# 第1位控制RS
if high:
tempData |= 0b00000001
else:
tempData &
= 0b11111110
return tempData
在向I2C发送数据前,根据配置的背光设置来决定背光控制位的值:
# 通过I2C总线写入数据
def writeI2C(addr, data):
# 添加背光控制
temp = addBlenControl(data)
# 写数据到I2C总线
BUS.write_byte(addr ,temp)
准备好了这些工具函数,我们只需要根据LCD1602的指令手册来设置具体的功能,发送要展示的数据即可,完整的代码如下:
#coding:utf-8
import time
import smbus
BUS = smbus.SMBus(1)
# LCD屏幕的硬件地址
LCD_ADDR = 0x27
# 是否开启背光 由PCF8574T的低4位中的第4位决定
BLEN = 1 # 补充背光控制位
def addBlenControl(data):
global BLEN
tmpData = https://www.songbingjia.com/android/data
if BLEN:
# 将第4位背光控制位强制设置1
tmpData = data | 0b00001000
else:
# 将第4位背光控制位强制设置为0
tmpData = data &
0b11110111
return tmpData# 补充Enable控制位
def addEnableControl(data, high):
tempData = data
# 第3位控制Enable
if high:
tempData |= 0b00000100
else:
tempData &
= 0b11111011
return tempData# 补充RS控制位
def addRSControl(data, high):
tempData = data
# 第1位控制RS
if high:
tempData |= 0b00000001
else:
tempData &
= 0b11111110
return tempData# 通过I2C总线写入数据
def writeI2C(addr, data):
# 添加背光控制
temp = addBlenControl(data)
# 写数据到I2C总线
BUS.write_byte(addr ,temp)# 发送指令到LCD1602
def sendCommand(comm):
# comm高4位数据传输
# 低4位先清空
buf = comm &
0b11110000
# 先将Enable置为高电平
buf = addEnableControl(buf, 1)
# 设置为指令模式
buf = addRSControl(buf, 0)
# 写入指令
writeI2C(LCD_ADDR ,buf)
time.sleep(0.002)
# 将Enable置为低电平 使产生低电平跳变来执行指令
buf = addEnableControl(buf, 0)
writeI2C(LCD_ADDR ,buf)# comm低4位数据传输
# 高4位先清空 并将低4位的数据移动到高4位
buf = (comm &
0b00001111) <
<
4
# 当次指令的低4位用来 做enable re rew的控制
# 先将Enable置为高电平
buf = addEnableControl(buf, 1)
writeI2C(LCD_ADDR ,buf)
time.sleep(0.002)
# 将Enable置为低电平 使产生低电平跳变来执行指令
buf = addEnableControl(buf, 0)
writeI2C(LCD_ADDR ,buf)# 发送数据到LCD
def sendData(data):
# data高4位数据传输
# 低4位先清空
buf = data &
0b11110000
# 先将Enable置为高电平
buf = addEnableControl(buf, 1)
# 设置为数据模式
buf = addRSControl(buf, 1)
writeI2C(LCD_ADDR ,buf)
time.sleep(0.002)
# 将Enable置为低电平 使产生低电平跳变来执行指令
buf = addEnableControl(buf, 0)
writeI2C(LCD_ADDR ,buf)# data低4位数据传输
buf = (data &
0b00001111) <
<
4
# 先将Enable置为高电平
buf = addEnableControl(buf, 1)
# 设置为数据模式
buf = addRSControl(buf, 1)
writeI2C(LCD_ADDR ,buf)
time.sleep(0.002)
# 将Enable置为低电平 使产生低电平跳变来执行指令
buf = addEnableControl(buf, 0)
writeI2C(LCD_ADDR ,buf)# 初始化方法
def initLCD():
# 启动时,LCD1602为8位模式 I2C传输数据时先传输的为低位数据
# 因此实际上的指令为 0b00100011
# 为指令6 将LCD1602设置为4位总线模式
sendCommand(0b00110010)
time.sleep(0.005)# 之后的指令都是4位总线模式
sendCommand(0b00110010)
time.sleep(0.005)
# 指令4 设置屏幕开启,无光标,无闪烁
sendCommand(0b00001100)
time.sleep(0.005)
# 指令1 清屏
sendCommand(0b00000001) # 设置屏幕要展示的文案 x,y确定位置
def printLCD(x, y, str):
# 2行 16 列
if x <
0:
x = 0
if x >
15:
x = 15
if y <
0:
y = 0
if y >
1:
y = 1
# 指令7 设置数据要展示的位置
addr = 0b10000000 + 0b00000100 * y + x
sendCommand(addr)
# 开始发送字符数据到LCD1602的数据寄存器
for chr in str:
# ord函数可以获取字符的ascil
sendData(ord(chr))# 主程序
initLCD()
printLCD(0, 0, /'Hello, world!\')
time.sleep(2)
sendCommand(0b00000001)
time.sleep(0.002)
printLCD(0, 0, \'Love China!\')
time.sleep(2)
sendCommand(0b00000001)
time.sleep(0.002)
printLCD(0, 0, \'Great Raspberry!\')
如上代码所示,所有的指令都采用的二进制的方式,便于对比指令手册进行理解。更多时候我们会采用十六进制数字来编写指令,这样会使代码看的干净很多。上面的示例代码是一个简单的应用程序,运行后可以直接在LCD屏幕上展示3句话:
Hello World!
Love China!
Great Raspberry!
其实此程序也是一个完整的LCD1602驱动,初始化完成后,我们可以通过其提供额sendData方法来实现各种各样的显示需求。效果如下图所示:
文章图片
推荐阅读
- Alibaba微服务技术系列「Dubbo3.0技术专题」回顾Dubbo2.x的技术原理和功能实现
- 关于静态代码块的执行顺序,很简单的一道题,应该所有人都会吧()
- vivo全球商城时光机 - 大型促销活动保障利器
- redis笔记01_李孟_新浪博客
- 如何理解虚拟机类加载过程详解
- 无法解开Visual Studio Code的方法,因为Apple无法在MacOS Catalina中检查它的恶意软件
- 如何解决Magento 2弃用问题(目录搜索当前已配置为使用已弃用的MySQL引擎。迁移到Elasticsearch引擎之一)
- 如何在Symfony 5中将对网站的访问限制为特定国家(地理封锁)
- 如何在Magento 2.3.2中重置测试销售/订单和仪表板信息