LCD1602液晶使用介绍–(完整版)

LCD1602液晶使用介绍–(完整版)lcd1602+c51介绍文章目录LCD1602介绍1602引脚信号说明控制器接口介绍1、基本操作时许2、状态字说明3、指令说明RAM地址映射控制时序图代码实现写入命令写数据试验例程CGRAM自定义字模(简易汉字显示)LCD1602介绍LCD1602液晶在实际的产品运用中也是比较多产品,应为前一段时间也正好用到了所以惊天就对LCD1602液晶做一个总结,方便以后阅读同时也希望能够帮住到需要的人,总结的可能存在错误欢迎指出!所谓的1602是指显示的时候,有2行内容每行有16个字符。其实这类字符型产

大家好,又见面了,我是你们的朋友全栈君。

lcd1602 +c51 介绍

LCD1602介绍

LCD1602液晶在实际的产品运用中也是比较多产品,应为前一段时间也正好用到了,所以今天就对LCD1602液晶做一个总结,方便以后阅读同时也希望能够帮住到需要的人,总结的可能存在错误欢迎指出!

所谓的1602是指显示的时候,有2行内容每行有16个字符。其实这类字符型产品都可以这样解读比如:lcd12864就是有128行64列。目前市面上字符液晶大多数是基于HD44780液晶芯片的,控制原理大多相同。因此基于HD44780写的液晶控制程序可以很方便适用于市面上大多数字符型液晶产品。
在这里插入图片描述

1602引脚信号说明

字符型LCD1602通常分为14条引脚和16引脚两种,16引脚多出来的是背光电源线VCC(15引脚)和地线(16引脚),其控制原理与14引脚LCD完全一样,引脚定义如下表:

引脚号 引脚名 电平 输入/输出 作用
1 Vss 电源地
2 Vcc 电源(+5V)
3 Vee 对比调整电压
4 RS 0/1 输入 0=输入指令;1=输出数据
5 R/W 0/1 输入 0=向LCD写入指令或数据; 1=从LCD读取信息
6 E 1,1->0 输入 使能信号,1时读取信息,1->0(下降沿)执行指令
7 DB0 0/1 输入/输出 数据总线line0(最低位)
8 DB1 0/1 输入/输出 数据总线line1
9 DB2 0/1 输入/输出 数据总线line2
10 DB3 0/1 输入/输出 数据总线line3
11 DB4 0/1 输入/输出 数据总线line4
12 DB5 0/1 输入/输出 数据总线line5
13 DB6 0/1 输入/输出 数据总线line6
14 DB7 0/1 输入/输出 数据总线line7
15 A +Vcc LCD背光灯源正极
16 K 接地 LCD背光灯源负极

控制器接口介绍

1、基本操作时许

  • 1.1读状态:输入:RS=L,RW=H,E=H
    —-输出:D0~D7=状态字
  • 1.2写指令:输入:RS=L,RW=L,D0~D7=指令码
    —-输出:无
  • 1.3读数据:输入:RS=H,RW=H,E=H
    —-输出:D0~D7=数据
  • 1.4写数据:输入:RS=H,RW=L,D0~D7=数据,E=高脉冲
    —-输出:无

2、状态字说明

STA7 D7 STA6 D6 STA5 D5 STA4 D4 STA3 D3 STA2 D2 STA1 D1 STA0 D0

STA0-6 当前数据地址指针的数值
STA7 读写操作使能 1:禁止 0:允许

对控制器每次进行读写操作前,都必须进行读写检测,确保STA7为0

3、指令说明

初始化设置
显示模式设置

在这里插入图片描述
显示开/关光标设置
在这里插入图片描述
在这里插入图片描述
数据控制
控制器内部没有一个数据地址指针,可以通过他们来访问内部的全部80字节RAM。
其他设置

指令码 功能
01H 显示清屏:1.数据指针清零 2.所有显示清零
02H 显示回车:1.数据指针清零

初始化过程:

延时
写指令38H
延时
写指令38H
延时
写指令38H
(每次写指令、读/写数据操作之前均需检测信号)
写指令38H:显示模式设置
写指令08H:显示关闭
写指令01H:显示清屏
写指令06H:显示光标移动设置
写指令0CH:显示开及光标设置

代码:

void LcdInit(){ 
   //LCD初始化程序
Delay1ms(15);
LcdWriteCom(0x38);
Delay1ms(5);
LcdWriteCom(0x38);
Delay1ms(5);
LcdWriteCom(0x38);//设置显示模式
LcdWriteCom(0x0C);//开显示不显示光标,光标不闪烁
LcdWriteCom(0x06);//写一个数据,指针加1
LcdWriteCom(0x01);//清屏
LcdWriteCom(0x80);//设置数据指针起点。
}

RAM地址映射

HD44780内置DDRAM、CGROM和CGRAM。
DDRAM就是显示数据RAM,用来寄存待显示的字符代码。共80个字节,地址和屏幕的对应关系如下:

显示位置 1 2 3 4 5 6 7 … … 40
第一行 00H 01H 02H 03H 04H 05H 06H … … 27H
第二行 40H 41H 42H 43H 44H 45H 46H … … 67H

也就是说想要在LCD1602屏幕上的第一行第一个位置显示一个“A”,就要向DDRAM的00H地址写“A”字的代码就OK了,但具体的写入是要按照LCD模块的指令格式来进行的。
但是我们发现每一行有40个地址,而我们们每行只能显示16个字符,其实际多的位置可以实现字符的移动,我们在看大佬作品的时候可能会见到有的字符是从左面移过来,他的实现形式就用到了着些多的地址。将数据先写到未显示的地址然后使用指令进行左移就可以了。
那么1602显示的地址又是什么呐?下图就是DDRAM地址与现实位置的对应关系。
LCD1602液晶使用介绍--(完整版)

控制时序图

1、读操作时序图:
在这里插入图片描述
2、写操作时序图:
在这里插入图片描述
3、时序参数
在这里插入图片描述

代码实现

LCDE = E //使能信号

写入命令

RS=L,RW=L,D0~D7=指令码,E=高脉冲。

void LcdWriteCom(unsigned char com){ 
   //写入命令
	RS = 0;
	RW = 0;
	GPIO_LCD=com;
	Delay1ms(10);
	LCDE = 1; //给一个高脉冲
	Delay1ms(10);
	LCDE = 0;
}

写数据

RS=H,RW=L,D0~D7=数据,E=高脉冲。

void LcdWriteData(unsigned char dat){ 
   //写入数据
	RS = 1;
	RW = 0;
	GPIO_LCD=dat;
	Delay1ms(10);
	LCDE = 1; //给一个高脉冲
	Delay1ms(10);
	LCDE = 0;

试验例程

main.c文件

#include<reg52.h>
#include"lcd.h"
unsigned char CnCh[] = "012345678912345";
unsigned char CnCh1[] = "ABCDEFGIJKLMNOP";
unsigned char i,a =0,j,n;
unsigned char code Data_1[]=" I Love You ";  // 第一行显示,共十六个字符
unsigned char code Data_2[]="Good Good Study,Day Day Up !";  // 第二行显示,共28个字符
unsigned char i;
void main(){ 

lcd_Init();
lcd_write_com(0x80);
for(i = 0;i<16;i++){ 

lcd_read_busy();
lcd_write_dat(CnCh[i]);
}
lcd_write_com(0xc0);
for(i = 0;i<16;i++){ 

lcd_read_busy();
lcd_write_dat(CnCh1[i]);
}
while(1);
}
void zimo(){ 

unsigned char code Data_0[]={ 
0x0f, 0x12, 0x0f, 0x0a, 0x1f, 0x02, 0x02, 0x02};  // 汉字 年 的字模
unsigned char code Data_1[]={ 
0x0f, 0x09, 0x0f, 0x09, 0x0f, 0x09, 0x0b, 0x11};  // 汉字 月 的字模 
unsigned char code Data_2[]={ 
0x1f, 0x11, 0x11, 0x1f, 0x11, 0x11, 0x1f, 0x00};  // 汉字 日 的字模
lcd_Init();  // LCD1602 初始化
lcd_write_com(0x40);  // 0100 0000; 指令 0x40 向 CGRAM 地址0 写入自定义数据
for(i=0; i<8; i++){ 

lcd_write_dat(Data_0[i]);  // 写入自定义字符字模
}
lcd_write_com(0x48);  // 0100 1000; 指令 0x48 向 CGRAM 地址1 写入自定义数据
for(i=0; i<8; i++){ 

lcd_write_dat(Data_1[i]);  // 写入自定义字符字模
}
lcd_write_com(0x50);  // 0101 0000; 指令 0x50 向 CGRAM 地址2 写入自定义数据
for(i=0; i<8; i++){ 

lcd_write_dat(Data_2[i]);  // 写入自定义字符字模
}
lcd_write_com(0x00 + 0x80);  // 在第一行第一列显示 第一个字符
lcd_write_dat(0);
lcd_write_com(0x02 + 0x80);  // 在第一行第三列显示 第二个字符
lcd_write_dat(1);
lcd_write_com(0x04 + 0x80);  // 在第一行第五列显示 第一个字符
lcd_write_dat(2);
while(1);
/* //分割线******************************************************************* unsigned char table[]={0x15,0x0A,0x15,0x0A,0x15,0x0A,0x15,0x0A}; lcd_Init(); lcd_write_com(0x40); for(i=0;i<8;i++){ lcd_write_dat(table[i]); } lcd_write_com(0x80); lcd_write_dat(0x00); while(1); */
}
//单行移动显示**************************************************************
void dh(){ 

lcd_Init();
lcd_write_com(0x80);  // 第一行第一列地址
for(i=0; i<16; i++){ 
     
lcd_write_dat(Data_1[i]);  // 显示第一行
}
while(1){ 

lcd_read_busy();
lcd_write_com(0xc0);  // 第二行第一列地址
for(j=n; j<28+n; j++){ 
     
lcd_write_dat(Data_2[j]);  // 显示第二行
}
n++;
if(n >= (28-15)){ 
  // 当数据移动到最后时,n 重置 0,停顿 500 ms,重新开始移动显示
n = 0;
delay1ms(50);
}
delay1ms(20);  // 控制移动速度
}
}

lcd.c文件

#include"lcd.h"
void delay1ms(unsigned char d)   //误差 0us
{ 

unsigned char a,b,c;
for(c=0;c<d;c++)
for(b=142;b>0;b--)
for(a=2;a>0;a--);
}
void lcd_Init(){ 

lcd_write_com(0x38);
delay1ms(1);
lcd_write_com(0x38);	//设置显示模式
lcd_write_com(0x0c);	//开显示不显示贯标
lcd_write_com(0x06);	//指针自动加一
lcd_write_com(0x01);	//清屏
lcd_write_com(0x80);	//设置数据指针起点
}
void lcd_write_com(unsigned char com){ 

RS = 0;
RW = 0;
E = 0;
P0 = com;
delay1ms(10);
E = 1;
delay1ms(10);
E = 0;
}
void lcd_write_dat(unsigned char dat){ 

RS = 1;
RW = 0;
E = 0;
P0 = dat;
delay1ms(10);
E = 1;
delay1ms(10);
E = 0; 
}
void lcd_xy(unsigned char x,unsigned char y){ 

lcd_write_com(x+y);
}
void lcd_read_busy(){ 

unsigned char he,a;
RS = 0;
RW = 1;
E = 1;
he = P0;
while(1){ 

he = P0;
E = 0;
if(!(he&0x80))//当he&0x80为0时跳出循环,表明不忙。
break;
E = 1;
a++;
if(a>=10)
break;
}
}

lcd.h文件

#ifndef __LCD_H_
#define __LCD_H_
#include<reg52.h>
sbit RS = P2^0;
sbit RW = P2^1;
sbit E = P2^2;
void delay1ms(unsigned char d);		//延时函数
void lcd_Init();		//初始化
void lcd_write_com(unsigned char com);		//写指令
void lcd_write_dat(unsigned char dat);		//写数据
void lcd_xy(unsigned char x,unsigned char y);  //写位置
void lcd_read_busy();		//检测标志位
#endif

CGRAM自定义字模(简易汉字显示)

这里说明一下lcd1602液晶是不能显示汉字的,因为它的显示原理是由若干个5X7或者5X11的点阵字符位组成的,又因为汉字较为复杂,所以1602的主要作用就是显示字母、数字、符号的。但是真的不能显示汉字吗?也并非绝对不能。接就是下面要说的CGRAM自定义字模。

要显示我们自定义的字符,就要用到LCD中的CGRAM存储器(character generate RAM),而我们之前用的显示自带的字符用到的是DDRAM,两个是不同的。看手册我们知道,CGRAM的容量是64个字节,而一个字符是8个字节,所以一共能显示8个自定义的字符。内部常用字符的显示是从0x20开始的,0x00 ~ 0x0F是专门留给自定义字符显示用的,0x00-0x07和0x08~0x0F显示的内容是一样的,也就是说0x00=0x08,0x01=0x09,以此类推。CGRAM共128个位,地址是0x40-0x7F,128/8=16正好对应的是0x00-0x0F共16个,刚才说了,0x00与0x08对应,0x01与0x09对应,共16个,这并不矛盾!说了这么多,那么怎样显示一个自定义字符呢?
首先我们要清楚LCD1602显示字符的点阵大小,眼力好的可以看出来,LCD1602一个显示字符的位置是58的点阵,也就是说它所能显示的点阵图形的大小是58的!要显示一个自定义的字符,首先我们要知道所显示自定义字符的点阵数据,也就是在一个58的点阵上那个点是黑的(将该点点黑,就是高电平—-1),哪个点是白的(该点不显示,为低电平—-0),但是我们送入到LCD中的是ASCII码,它是8位的数据,而一个显示字符的点阵大小只是58的,显然不够,显示的办法是8*8点阵的前三列不用,也就是不显示,我们只用后面的5列来显示。
然后设定我们是要定义第几个自定义字符,前面已经介绍了,LCD1602最多显示8个自定义字符;然后要规定在液晶的什么位置显示自定义字符,看过数据手册我们知道,第一行第一个位置的地址是0x80,第二行一个位置的地址是0xC0。最后就是要显示我们定义的第几个字符其对应CGRAM地址的关系式是:

0x00:第一个(0x40) 0x01:第二个(0x48)
0x02:第三个(0x50) 0x03:第四个(0x58)
0x04:第五个(0x60) 0x05:第六个(0x68)
0x06:第七个(0x70) 0x07:第八个(0x78)

每个字符由5X8点阵组成(也可选用5X10) ,想要实现显示,只需如下图:
例:以5X8点阵为例,显示字符 A

0代表灭,1代表亮
只需将想要显示的字符的对应位置1,就能显示该字符

01110	□█ █ █□
10001	█ □□□ █
10001	█ □□□ █
10001	█ □□□ █
11111	█ ███ █
10001	█ □□□ █
10001	█ □□□ █
10001	█ □□□ █
A={0x0e,0x11,0x11,0x11,0x1f,0x11,0x11,0x11}

代码
流程

首先创建自定义字模
把字模存入CGRAM中对应的自定义位置,位置在上表中以显示出来了
在LCD1602中显示出来

void type_model_diy(){ 

unsigned char code Data_0[]={ 
0x0e,0x11,0x11,0x11,0x1f,0x11,0x11,0x11};  // 字母A
lcd_Init();  // LCD1602 初始化
lcd_write_com(0x40);  // 0100 0000; 指令 0x40 向 CGRAM 地址0 写入自定义数据
for(i=0; i<8; i++){ 

lcd_write_dat(Data_0[i]);  // 写入自定义字符字模
}
lcd_write_com(0x00 + 0x80);  // 在第一行第一列显示 第一个字符
lcd_write_dat(0);
while(1);
}

END!
在这里插入图片描述

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/160148.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)
blank

相关推荐

  • 微信不能登录网页版(微信手机网页登录)

     因为出于工作和学习的目的,我的个人电脑操作系统使用的是Ubuntu18.04LTS,就目前而言,许多优秀的软件都有Linux版本,虽然Linux的用户相对群体较小,但是其软件生态也在逐渐成长,而且日常使用浏览器就可以解决许多的应用需求。现在微信和QQ在生活和工作领域均是举足轻重,emm…  BUT!!!腾讯到目前为止并没有推出Linux版的微信和TI…

  • CSS控制文本超出指定宽度显示省略号和文本不换行

    一般的文字截断(适用于内联与块):.text-overflow{display:block;/*内联对象需加*/width:31em;/*何问起hovertree.com*/word

    2021年12月23日
  • mapminmax 用法

    mapminmax 用法mapminmax是MATLAB实现归一化的工具包,默认:(1)将矩阵的每行分别进行归一化;(2)每行的最大值最小值作为每行归一化的xmin和xmax;(3)将数据归一化到[-1,1].若要将数据归一化到0到1之间,即y∈[0,1],使用b=mapminmax(a,0,1);若给与确定的最大值和最小值作为每行的xmin和xmax,使用:b= mapminmax(a,0,1);PS.xmin…

  • 我倒在了美团面试算法题:字符串大数相加

    我倒在了美团面试算法题:字符串大数相加

    2020年11月20日
  • 三星note3怎样刷原生Android,三星note三可以刷原生android系统吗?

    三星note3怎样刷原生Android,三星note三可以刷原生android系统吗?可以的。1刷之前要备份好个人的通讯录等资料。如果你的手机使用正常就不用去刷了。自己刷也是可以的,但要到网上下载手机软件,三星的网上版本多,有些是专为水货编写的。2刷机最好在风险可控前提下的刷机。当前DIY的版本都是基于原版的,只不过是将原来的图片替换成另外的图片,将原来的铃声替换成另外的铃声,没有动核心部分。只是替换更改了部分图片、铃声或者菜单字符等,所以不应该有不良影响,按步骤操作,刷机是基本上…

  • Python的八种数据类型[通俗易懂]

    Python的八种数据类型[通俗易懂]Python的八种数据类型八种数据类型分别是:number(数字)、string(字符串)、Boolean(布尔值)、None(空值)list(列表)、tuple(元组)、dict(字典)、se

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号