单片机显示原理(LCD1602)

2023-05-16

一.接口

 LCD1602是很多单片机爱好者较早接触的字符型液晶显示器,它的主控芯片是HD44780或者其它兼容芯片。与此相仿的是LCD12864液晶显示器,它是一种图形点阵显示器,能显示的内容比LCD1602要丰富得多,除了普通字符外,还可以显示点阵图案,带有汉字库的还可以显示汉字,它的并行驱动方式与LCD1602相差无几,所以,在这里花点时间是值得的。

一般来说,LCD1602有16条引脚,据说还有14条引脚的,与16脚的相比缺少了背光电源A(15脚)和地线K(16脚)。这块LCD1602的型号是HJ1602A,是绘晶科技公司的产品,它有16条引脚。如图1所示:



图1

再来一张它的背面的,如图2所示:



图2

它的16条引脚定义如下:

引脚号

符号

引脚说明

引脚号

符号

引脚说明

1

VSS

电源地

9

D2

数据端口

2

VDD

电源正极

10

D3

数据端口

3

VO

偏压信号

11

D4

数据端口

4

RS

命令/数据

12

D5

数据端口

5

RW

读/写

13

D6

数据端口

6

E

使能

14

D7

数据端口

7

D0

数据端口

15

A

背光正极

8

D1

数据端口

16

K

背光负极

对这个表的说明:

1.    VSS接电源地。

2.    VDD接+5V。

3.    VO是液晶显示的偏压信号,可接10K的3296精密电位器。或同样阻值的RM065/RM063蓝白可调电阻。见图3。


图3

4.    RS是命令/数据选择引脚,接单片机的一个I/O,当RS为低电平时,选择命令;当RS为高电平时,选择数据。

5.    RW是读/写选择引脚,接单片机的一个I/O,当RW为低电平时,向LCD1602写入命令或数据;当RW为高电平时,从LCD1602读取状态或数据。如果不需要进行读取操作,可以直接将其接VSS。

6.    E,执行命令的使能引脚,接单片机的一个I/O。

7.    D0—D7,并行数据输入/输出引脚,可接单片机的P0—P3任意的8个I/O口。如果接P0口,P0口应该接4.7K—10K的上拉电阻。如果是4线并行驱动,只须接4个I/O口。

8.    A背光正极,可接一个10—47欧的限流电阻到VDD。

9.    K背光负极,接VSS。见图4所示。


图4

二.基本操作

LCD1602的基本操作分为四种:

1.    读状态:输入RS=0,RW=1,E=高脉冲。输出:D0—D7为状态字。

2.    读数据:输入RS=1,RW=1,E=高脉冲。输出:D0—D7为数据。

3.    写命令:输入RS=0,RW=0,E=高脉冲。输出:无。

4.    写数据:输入RS=1,RW=0,E=高脉冲。输出:无。

读操作时序图(如图5):


 图5

写操作时序图(如图6):


图6

时序时间参数(如图7):


图7

三.DDRAM、CGROM和CGRAM

DDRAM(Display Data RAM)就是显示数据RAM,用来寄存待显示的字符代码。共80个字节,其地址和屏幕的对应关系如下(如图8):


 图8

DDRAM相当于计算机的显存,我们为了在屏幕上显示字符,就把字符代码送入显存,这样该字符就可以显示在屏幕上了。同样LCD1602共有80个字节的显存,即DDRAM。但LCD1602的显示屏幕只有16×2大小,因此,并不是所有写入DDRAM的字符代码都能在屏幕上显示出来,只有写在上图所示范围内的字符才可以显示出来,写在范围外的字符不能显示出来。这样,我们在程序中可以利用下面的“光标或显示移动指令”使字符慢慢移动到可见的显示范围内,看到字符的移动效果。

前面说了,为了在液晶屏幕上显示字符,就把字符代码送入DDRAM。例如,如果想在屏幕左上角显示字符‘A’,那么就把字符‘A’的字符代码41H写入DDRAM的00H地址处即可。至于怎么写入,后面会有说明。那么为什么把字符代码写入DDRAM,就可以在相应位置显示这个代码的字符呢?我们知道,LCD1602是一种字符点阵显示器,为了显示一种字符的字形,必须要有这个字符的字模数据,什么叫字符的字模数据,看看下面的这个图就明白了(如图9)。


图9

上图的左边就是字符‘A’的字模数据,右边就是将左边数据用“○”代表0,用“■”代表1。从而显示出‘A’这个字形。从下面的图可以看出,字符‘A’的高4位是0100,低4位是0001,合在一起就是01000001b,即41H。它恰好与该字符的ASCII码一致,这样就给了我们很大的方便,我们可以在PC上使用P2=‘A’这样的语法。编译后,正好是这个字符的字符代码。

在LCD1602模块上固化了字模存储器,就是CGROM和CGRAM,HD44780内置了192个常用字符的字模,存于字符产生器CGROM(Character Generator ROM)中,另外还有8个允许用户自定义的字符产生RAM,称为CGRAM(Character Generator RAM)。下图(如图12)说明了CGROM和CGRAM与字符的对应关系。从ROM和RAM的名字我们也可以知道,ROM是早已固化在LCD1602模块中的,只能读取;而RAM是可读写的。也就是说,如果只需要在屏幕上显示已存在于CGROM中的字符,那么只须在DDRAM中写入它的字符代码就可以了;但如果要显示CGROM中没有的字符,比如摄氏温标的符号,那么就只有先在CGRAM中定义,然后再在DDRAM中写入这个自定义字符的字符代码即可。和CGROM中固化的字符不同,CGRAM中本身没有字符,所以要在DDRAM中写入某个CGROM不存在的字符,必须在CGRAM中先定义后使用。程序退出后CGRAM中定义的字符也不复存在,下次使用时,必须重新定义。


图10

上面这个图(如图10)说明的是5×8点阵和5×10点阵字符的字形和光标的位置。先来说5×8点阵,它有8行5列。那么定义这样一个字符需要8个字节,每个字节的前3个位没有被使用。例如,定义摄氏温标的符号{0x10,0x06,0x09,0x08,0x08,0x09,0x06,0x00}。


 图11

上面这个图(如图11)说明的是设置CGRAM地址指令。从这个指令的格式中我们可以看出,它共有aaaaaa这6位,一共可以表示64个地址,即64个字节。一个5×8点阵字符共占用8个字节,那么这64个字节一共可以自定义8个字符。也就是说,上面这个图的6位地址中的DB5DB4DB3用来表示8个自定义的字符,DB2DB1DB0用来表示每个字符的8个字节。这DB5DB4DB3所表示的8个自定义字符(0--7)就是要写入DDRAM中的字符代码。我们知道,在CGRAM中只能定义8个自定义字符,也就是只有0—7这8个字符代码,但在下面的这个表(如图12)中一共有16个字符代码(××××0000b--××××1111b)。实际上,如图所示,它只能表示8个自定义字符 (××××0000b=××××1000b, ××××0001b=××××1001b……依次类推)。也就是说,写入DDRAM中的字符代码0和字符代码8是同一个自定义字符。 5×10点阵每个字符共占用16个字节的空间,所以CGRAM中只能定义4个这样的自定义字符。

那么如何在CGRAM中自定义字符呢?在上面的介绍中,我们知道有一个设置CGRAM地址指令,同写DDRAM指令相似,只须设置好某个自定义字符的字模数据,然后按照上面介绍的方法,设置好CGRAM地址,依次写入这个字模数据即可。我们在后面的例子中再进行说明。


 图12

四.LCD1602指令

1.工作方式设置指令(如图13)


 图13

×:不关心,也就是说这个位是0或1都可以,一般取0。

DL:设置数据接口位数。

DL=1:8位数据接口(D7—D0)。

DL=0:4位数据接口(D7—D4)。

N=0:一行显示。

N=1:两行显示。

F=0:5×8点阵字符。

F=1:5×10点阵字符。

说明:因为是写指令字,所以RS和RW都是0。LCD1602只能用并行方式驱动,不能用串行方式驱动。而并行方式又可以选择8位数据接口或4位数据接口。这里我们选择8位数据接口(D7—D0)。我们的设置是8位数据接口,两行显示,5×8点阵,即0b00111000也就是0x38。(注意:NF是10或11的效果是一样的,都是两行5×8点阵。因为它不能以两行5×10点阵方式进行显示,换句话说,这里用0x38或0x3c是一样的)。

2.显示开关控制指令(如图14)


图14

D=1:显示开,D=0:显示关。

C=1:光标显示,C=0:光标不显示。

B=1:光标闪烁,B=0:光标不闪烁。

说明:这里的设置是显示开,不显示光标,光标不闪烁,设置字为0x0c。

3.进入模式设置指令(如图15、16)


图15

I/D=1:写入新数据后光标右移。

I/D=0:写入新数据后光标左移。

S=1:显示移动。

S=0:显示不移动。


 图16

说明:这里的设置是0x06。

4.光标或显示移动指令(如图17、18)


图17


 图18

说明:在需要进行整屏移动时,这个指令非常有用,可以实现屏幕的滚动显示效果。初始化时不使用这个指令。

5.清屏指令(如图19)


图19

说明:清除屏幕显示内容。光标返回屏幕左上角。执行这个指令时需要一定时间。

6.光标归位指令(如图20)

 

 图20

说明:光标返回屏幕左上角,它不改变屏幕显示内容。

7.设置CGRAM地址指令(如图21)


图21

说明:这个指令在上面已经介绍过。用法在后面例子中说明。

8.设置DDRAM地址指令(如图22)


图22

说明:这个指令用于设置DDRAM地址。在对DDRAM进行读写之前,首先要设置DDRAM地址,然后才能进行读写。前面我们说过,DDRAM就是LCD1602的显示存储器。我们要在它上面进行显示,就要把要显示的字符写入DDRAM。同样,我们想知道DDRAM某个地址上有什么字符,也要先设置DDRAM地址,然后将它读出到单片机。

9.读忙信号和地址计数器AC(如图23)


图23

说明:这个指令用来读取LCD1602状态。对于单片机来说,LCD1602属于慢速设备。当单片机向其发送一个指令后,它将去执行这个指令。这时如果单片机再次发送下一条指令,由于LCD1602速度较慢,前一条指令还未执行完毕,它将不接受这新的指令,导致新的指令丢失。因此这条读忙指令可以用来判断LCD1602是否忙,能否接收单片机发来的指令。当BF=1,表示LCD1602正忙,不能接受单片机的指令;当BF=0,表示LCD1602空闲,可以接收单片机的指令。RS=0,表示是指令;RW=1,表示是读取。这条指令还有一个副产品:即可以得到地址记数器AC的值(address counter)。LCD1602维护了一个地址计数器AC,用来记录下一次读写CGRAM或DDRAM的位置。需要强调的是:这条指令我一次也没有执行成功。很多网友似乎也是这样。好在我们有另外的办法,也就是延时。通过查看每条指令的执行时间,再经过一些试验,可以确定指令的延时。这样就可以在上一条指令执行完毕后再执行下一条指令了。

10.写数据到CGRAM或DDRAM指令(如图24)


图24

说明:RS=1,数据;RW=0,写。指令执行时,要在DB7—DB0上先设置好要写入的数据,然后执行写命令。

11.从CGRAM或DDRAM读数据指令(如图25)


图25

说明:RS=1,数据;RW=1,读。先设置好CGRAM或DDRAM的地址,然后执行读取命令。数据就被读入后DB7—DB0。

 

五.实例

下面我们就以一个实例来结束这篇文章。先介绍一下背景:单片机最小系统(扩充了外部RAM 62256)。采用STC89C52RC,晶振22.1184MHZ。以5×8点阵,16×2行,8位数据端口。首先在第一行显示“I love MCU!”,第二行显示“LCD1602 Test!”。延时一段时间,清屏。然后在第一行显示自定义字符:摄氏温标标志。第二行显示圆周率(pai)标志。再延时一段时间,清屏。最后在第一行显示“Welcome to my blog!”,显示方式是从屏幕右面移入,左面移出。周而复始(如图26)。


图26


  1 #ifndef __ZHANGTYPE_H__
  2 #define __ZHANGTYPE_H__
  3  
  4 #define uint8    unsigned char
  5 #define uint16   unsigned short int
  6 #define uint32   unsigned long int
  7 #define int8     signed char
  8 #define int16    signed short int
  9 #define int32    signed long int
 10 #define uint64   unsigned long long int
 11 #define int64    signed long long int
 12  
 13 #endif 
 14 //File2
 15  
 16 #ifndef __FUN_H__
 17 #define __FUN_H__ 
 18 #include "ZhangType.h"
 19 #include  
 20 void Delay(uint16 time); 
 21 #endif 
 22 //File3
 23  
 24 #include "fun.h" 
 25 void Delay(uint16 time)
 26 {
 27     while(time--);
 28 } 
 29 //File4
 30  
 31 #ifndef __1602_H__
 32 #define __1602_H__
 33  
 34 #include
 35 #include "ZhangType.h"          //变量类型
 36 #include "fun.h"            //常用函数 
 37  
 38 #define    SETMODE    0x38          //16*2显示,5*7点阵,8位数据接口
 39 #define DISOPEN   0x0C          //显示开,不显示光标,光标不闪烁
 40 #define DISMODE   0x06          //读写字符后地址加1,屏显不移动
 41 #define    SETADDR    0x80          //设置数据地址指针初始值
 42 #define CLEAR 0x01          //清屏,数据指针清零
 43 #define RET       0x02          //回车,数据指针清零 
 44 #define PORT  P2            //I/O口 
 45  
 46 sbit RS = P1^0;
 47 sbit RW = P1^1;
 48 sbit E = P1^2; 
 49  
 50 void Init1602(void);        //初始化1602
 51 void Write1602_Com(uint8 com);  //写命令
 52 void Write1602_Dat(uint8 dat);  //写数据
 53 void CheckBusy(void);           //检查忙
 54 void Write1602_One_Dat(uint8 X,uint8 Y,uint8 dat);          //写一个数据
 55 void Write1602_Str(uint8 addr,uint8 length,uint8 *pbuf);    //写一个数据串 
 56 #endif//
 57 //File5
 58  
 59 #include "1602.h" 
 60  
 61 void Write1602_Com(uint8 com)
 62 {
 63     E=0;
 64     RS=0;                    //命令
 65     Delay(50);               //延时
 66     RW=0;                    //
 67     Delay(50);
 68     PORT=com;                //端口赋值
 69     Delay(50);
 70     E=1;                     //高脉冲
 71     Delay(50);
 72     E=0;
 73 }
 74  
 75 void Write1602_Dat(uint8 dat)
 76 {
 77     E=0;
 78     RS=1;                    //数据
 79     Delay(50);               //延时
 80     RW=0;                    //
 81     Delay(50);
 82     PORT=dat;                //端口赋值
 83     Delay(50);
 84     E=1;                     //高脉冲
 85     Delay(50);
 86     E=0;
 87 } 
 88  
 89 void CheckBusy(void)
 90 {
 91     uint8 temp;
 92     RS=0;                    //命令
 93     RW=1;                    //
 94     E=0;
 95     while(1)
 96     {
 97        PORT=0xFF;           //端口为输入
 98        E=1;                 //高脉冲
 99        temp=PORT;
100        E=0;
101        if ((temp&0x80)==0)      //检查BF位是否为0
102            break;
103     }
104 } 
105  
106 void Init1602(void)
107 {
108     Write1602_Com(SETMODE);     //模式设置
109     Delay(500);
110     Write1602_Com(DISOPEN);     //显示设置
111     Delay(500);
112     Write1602_Com(DISMODE);     //显示模式
113     Delay(500);
114     Write1602_Com(CLEAR);       //清屏
115     Delay(500);
116 } 
117  
118 void Write1602_One_Dat(uint8 x,uint8 y,uint8 dat)
119 {
120     x&=0x0f;
121     y&=0x01;
122     if(y)
123        x|=0x40;
124     x|=0x80;
125     Write1602_Com(x);
126     Write1602_Dat(dat);
127 } 
128  
129 void Write1602_Str(uint8 addr,uint8 length,uint8 *pbuf)
130 {
131     uint8 i; 
132     Write1602_Com(addr);
133     for(i=0;i
134     {
135        Write1602_Dat(pbuf[i]);
136     }
137 } 
138 //File6
139 *******************************************************
140 *名称:主文件(_main.c)
141 *功能:测试
142 *日期:2014/09/09
143 *******************************************************/
144 #include "1602.h"
145 #include "fun.h" 
146 uint8 code hot[8]={                                  //摄氏温度字模
147 0x10,0x06,0x09,0x08,0x08,0x09,0x06,0x00
148 };
149 uint8 code pi[8]={                       
150 0x00,0x1f,0x0a,0x0a,0x0a,0x13,0x00,0x00                 //pai
151 }; 
152 uint8 code strMCU[]="I love MCU!";
153 uint8 code strTest[]="LCD1602 Test!";
154 uint8 code blog[]="Welcome to my blog!"; 
155 uint8 i; 
156 void main()
157 {
158     Init1602();                                      //初始化1602 
159     //自定义CGRAM
160     Write1602_Str(0x40,8,hot);                       //摄氏温标
161     Write1602_Str(0x48,8,pi);                        //pai
162  
163     Write1602_Str(0x80,strlen(strMCU),strMCU);           //"I love MCU!"
164     Write1602_Str(0x80+0x40,strlen(strTest),strTest);    //"LCD1602 Test!"
165  
166     for(i=0;i<50;i++)                             //延时一段时间
167        Delay(10000);
168  
169     Write1602_Com(CLEAR);                            //指令执行时间较长
170     Delay(500);                                      //多加一些延时
171     for(i=0;i<16;i++)
172        Write1602_Dat(0);
173    
174     Write1602_Com(0xc0);                          //设置DDRAM地址
175     for(i=0;i<16;i++)
176        Write1602_Dat(1);
177     for(i=0;i<50;i++)                             //延时一段时间
178        Delay(10000);
179    
180     Write1602_Com(CLEAR);                            //指令执行时间较长
181     Delay(500);                                      //多加一些延时
182     Write1602_Str(0x80+0x10,strlen(blog),blog);          //写在显示之外
183     while(1)
184     {
185        Write1602_Com(0x18);                      //左移
186        for(i=0;i<20;i++)                         //延时
187            Delay(10000);
188     }
189 } 
190 //############################# THE END#############################  
View Code

 

转载于:https://www.cnblogs.com/hui088/p/4732034.html

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

单片机显示原理(LCD1602) 的相关文章

  • TeamViewer 的替代品 ZeroTier + NoMachine

    之前不怎么用 TeamViewer xff0c 最近用的多了 xff0c 特别好用 xff0c 有点上瘾 xff0c 在哪儿都能连家里的 RTX xff0c 太棒了 然后它就开始作了 有没有好的替代方案呢 xff1f 有人推荐向日葵 xff
  • Kubernetes tutorial - K8S 官方入门教程 中文翻译

    官方教程 xff0c 共 6 个小节 每一小节的第一部分是知识讲解 xff0c 第二部分是在线测试环境的入口 kubectl 的命令手册 原文地址 1 创建集群 1 1 使用 Minikube 创建集群 Kubernetes 集群 Kube
  • 两年,从纯小白到腾讯阿里,应届非计算机毕业生的2014互联网求职之路

    0 写在前面 以此文 xff0c 献给自己这两年的青葱岁月 xff0c 感谢淘宝的朗英师兄和微博的旭爷 全栈pm莹姐姐 酷炫石女王 以及我逝去的头发 参照Lucida的 9个offer xff0c 12家公司 xff0c 35场面试 一文
  • 在cmd中使用start运行exe文件闪退问题

    如果我们是使用的bat批处理文件来执行某件事 xff0c 我们可以直接使用 k参数 xff0c 或者pause命令来让容器 cmd 执行完文件后不退出 如果我们是在cmd中使用start或其他命令直接执行任务 xff0c 导致当前容器退出
  • GreenPlum 5.0的安装

    基本环境 xff1a serverIPMDW172 16 16 31SDW1172 16 16 34SDW2172 16 16 35 1 xff1a 检查操作系统是否符合要求 xff0c 以及系统设置 我这里使用的系统是CentOS 操作系
  • 转载:Pixhawk源码笔记八:添加新的参数

    转载 xff1a 新浪 64 WalkAnt 第九部分 添加新的参数 英文参考 xff1a http dev ardupilot com wiki code overview adding a new parameter 本节源自 xff1
  • Ubuntu(Linux) 下 unzip 命令使用详解

    1 功能作用 xff1a 解压缩zip文件 2 位置 xff1a usr bin unzip 3 格式用法 xff1a unzip Z opts modifiers file zip list x xlist d exdir 4 主要参数
  • Linux之Libcurl库的介绍与应用20170509

    一 LibCurl简介 LibCurl是免费的客户端URL传输库 xff0c 支持FTP FTPS HTTP HTTPS SCP SFTP TFTP TELNET DICT FILE xff0c LDAP等协议 xff0c 其主页是http
  • 面试问题之操作系统:动态链接库和静态链接库的区别

    动态链接库是一个可以被其它应用程序共享的程序模块 xff0c 其中封装了一些可以被共享的例程和资源 动态链接库文件名的扩展名一般是dll xff0c 也有可能是drv xff0c sys和fon xff0c 它和可执行文件 exe 非常类似
  • (uC/OS-II学习笔记) 消息邮箱&&消息队列

    原文出处 xff1a http www cnblogs com hebaichuanyeah 与信号量一样 xff0c 消息邮箱与消息列队都是一种事件块 消息邮箱可以在任务间实现信息传递 比如 xff0c 在任务1中发送一条消息 xff0c
  • rosbag使用--记录深度相机数据

    首先看一下教程 xff1a http wiki ros org openni launch Tutorials BagRecordingPlayback 知道了rosbag如何进行使用记录深度数据 但是按照以上教程记录下来的bag file
  • Postman 基本操作学习

    History 所有使用postman发送的request都会保存在这里 点击之后会在当前Tab打开 参考 xff1a Requests History Environments 这里用来设定当前request 发送时使用的环境 xff0c
  • linux内核去掉设备驱动,基于嵌入式Linux内核的系统设备驱动程序开发设计

    引言 Linux是一个遵循POSIX标准的免费操作系统 具有BSD和SYSV的扩展特性 与其他操作系统相比 xff0c 嵌入式Linux系统以其可应用于多种硬件平台 内核高效稳定 源码开放 软件丰富 网络通信和文件管理机制完善等优良特性而正
  • html onmouseover 注释掉,HTML onmouseover事件用法及代码示例

    将鼠标指针移到元素或其子元素上时 xff0c 将发生HTML中的DOM onmouseover事件 用法 在HTML中 xff1a 在JavaScript中 xff1a object onmouseover 61 function mySc
  • stylegan3自己导出的requirements.txt环境文件

    stylegan3官方给的环境文件是 environment yml xff0c 需要用conda安装 在linux上安装总会遇见奇奇怪怪的问题 xff0c 因此我导出requirements txt文件 xff0c 用pip安装 requ
  • html5 在线摄像头,HTML5在线摄像头使用

    HTML5在线摄像头应用 最近在搞一个考试系统 xff0c 系统要求要有随机拍照的功能 xff0c 并且摄像头能够收到js的控制 在线摄像头嘛 xff0c 就那两种实现的方式 xff1a cab或者flash 暂且不论本人从没学过的flas
  • STM32固件库文件编程结构思想的理解

    STM32的固件库文件功能相当完善 xff0c 提供的API完全能满足一般的项目需要 刚从51单片机转到STM32的人 xff0c 肯定会被这么庞大的东东吓到 xff0c 51单片机上对IO口操作 xff0c 简简单单几行代码就搞定了 xf
  • ekf pose使用方法 ros_【ROS-Gazebo】为什么选择SDF?

    前言 这是一个系列小文章 xff0c 主要介绍在ROS Gazebo中如何更好地使用SDF格式建模与仿真 众所周知 xff0c URDF是ROS的原生支持格式 xff0c 但在某些情况下 xff08 尤其是Gazebo仿真时 xff09 x
  • nvidia-smi命令执行很慢,如何改进

    初次安装好nvidia的驱动 xff0c 每次执行nvidia smi命令时 xff0c 要5秒以上 可通过如下命令进行改进 xff1a nvidia persistenced persistence mode 转载于 https www
  • 构建千万用户级别后台数据库架构

    关于如何构建千万级别用户的后台数据库架构话题 xff0c 在ITPUB及CSDN论坛都有不少网友提问 xff0c 新型问答网站知乎上也有人提问 xff0c 并且顺带梳理了下思路 xff0c 方便更多的技术朋友有章可循 xff0c 整理一篇抛

随机推荐