STC89C52 使用 LCD 1602
基本概述
引脚说明
- 第 1 脚: VSS 为电源地
- 第 2 脚: VDD 接 5V 正电源
- 第 3 脚: VL 为液晶显示器对比度调整端,接正电源时对比度最弱,接地时对比度最高,对比度过高时会产生“鬼影”,使用时可以通过一个 10K 的电位器调整对比度。
- 第 4 脚:RS 为寄存器选择,高电平时选择数据寄存器、低电平时选择指令寄存器。
- 第 5 脚:R/W 为读写信号线,高电平时进行读操作,低电平时进行写操作。当 RS 和 R/W 共同为低电平时可以写入指令或者显示地址,当 RS 为低电平 R/W 为高电平时可以读忙信号,当 RS 为高电平 R/W 为低电平时可以写入数据。
- 第 6 脚:E 端为使能端,当 E 端由高电平跳变成低电平时,液晶模块执行命令。
- 第 7-14 脚:D0~D7 为 8 位双向数据线。
- 第 15 脚:背光源正极。
- 第 16 脚:背光源负极。
模块控制指令
开发逻辑
- 在哪写?
- 写什么?
时序参数
写时序图
分析:
- RS : 高电平时为数据寄存器,低电平时为指令寄存器。 1 或 0 。
- RW : 0 低电平为写操作
- E :0 低电平 -> 到高电平,间隙时间 tps1 最小为 30 ns。
- DB0-DB7 : 开始写入数据,持续时间 tpw 最小值为 25 ns。
- E : 高电平 -> 低电平,其中间隙时间 tpw + tf 。
代码如下:
#include "reg52.h"
#include "intrins.h"
/*
RS -- P1.0
RW -- P1.1
E -- P1.4
*/
#define databuffer P0 //定义8位数据线,Po端口组
// 寄存器选择位 , 高电平选择【数据寄存器】,低电平选择【指令寄存器】
sbit RS = P1^0;
// 读写控制位 ,高电平为【读操作】,低电平为【写操作】。
sbit RW = P1^1;
// 使能型号
sbit E = P1^4;
/*
写指令
*/
void Write_Cmd_Func(char cmd )
{
RS = 0 ; // 寄存器选择
RW = 0 ; // 读写寄存器
E = 0 ;
_nop_();
E = 1 ;
databuffer = cmd ;
_nop_();
E= 0 ;
}
/*
写数据
*/
void Write_Data_Func(char dt)
{
RS = 1 ; // 寄存器选择
RW = 0 ; // 读写寄存器
E = 0 ;
_nop_();
E = 1 ;
databuffer = dt ;
_nop_();
E= 0 ;
}
读时序图
分析和上面类似,唯一的区别就是读取数据的时间点变化。
/*
读取指令
*/
char Read_Cmd_Func()
{
char tmp = NULL;
RS = 0 ;
RW = 1;
E = 0 ;
_nop_();
E = 1 ;
_nop_();
tmp = databuffer ;
E = 0;
_nop_();
return tmp ;
}
/*
读取数据
*/
char Read_Data_Func()
{
char tmp = NULL;
RS = 1 ;
RW = 1;
E = 0 ;
_nop_();
E = 1 ;
_nop_();
tmp = databuffer ;
E = 0;
_nop_();
return tmp ;
}
好的我们已经学会了,lcd1602的读写,那么下面实操。
写入字符
如何确定位置?
必定事手册写出来了的。
当然这里是不能直接只用的,手册对于其写入位置规定如下:
例如 第二行第一个字符的地址是 40H,那么是否直接写入 40H 就可以将光标定位在第二行第 一个字符的位置呢?这样不行,因为写入显示地址时要求最高位 D7 恒定为高电平 1 所以实际写入的数据应该是 01000000B(40H)+10000000B(80H)=11000000B(C0H)。
如何确定内容?
lcd 1602 所提供的字库表和C语言中的编码刚好一致,所以不需要写二进制数,直接写字符即可。
知道了写哪里,怎么写那就看看下面的代码实现。
代码实现
#include "reg52.h"
#include "intrins.h"
/*
RS -- P1.0
RW -- P1.1
E -- P1.4
*/
#define databuffer P0 //定义8位数据线,Po端口组
// 寄存器选择位 , 高电平选择【数据寄存器】,低电平选择【指令寄存器】
sbit RS = P1^0;
// 读写控制位 ,高电平为【读操作】,低电平为【写操作】。
sbit RW = P1^1;
// 使能型号
sbit E = P1^4;
/*
测忙:
手册中要求每次读写数据需要测忙 ,其标准 D7 位是否为高电平
其实就是读取数据。根据时序图进行编写 , 这里不够精准,可以自行修改
*/
void check_buys()
{
char tmp = 0x80;
databuffer = 0x80;
while(tmp & 0x80){
RS = 0 ;
RW = 1;
E = 0 ;
_nop_();
E = 1 ;
_nop_();
tmp = databuffer ;
E = 0;
_nop_();
}
}
/*
写指令
根据时序图进行编写 , 这里不够精准,可以自行修改
*/
void Write_Cmd_Func(char cmd )
{
check_buys();
RS = 0 ; // 寄存器选择
RW = 0 ; // 读写寄存器
E = 0 ;
_nop_();
E = 1 ;
databuffer = cmd ;
_nop_();
E= 0 ;
}
/*
写数据
根据时序图进行编写 , 这里不够精准,可以自行修改
*/
void Write_Data_Func(char dt)
{
check_buys();
RS = 1 ; // 寄存器选择
RW = 0 ; // 读写寄存器
E = 0 ;
_nop_();
E = 1 ;
databuffer = dt ;
_nop_();
E= 0 ;
}
void Delay15ms() //@11.0592MHz
{
unsigned char i, j;
i = 27;
j = 226;
do
{
while (--j);
} while (--i);
}
void Delay5ms() //@11.0592MHz
{
unsigned char i, j;
i = 9;
j = 244;
do
{
while (--j);
} while (--i);
}
/*
初始化
根据提供的手册编写即可
*/
void LCD1602_INIT()
{
//(1)延时 15ms
Delay15ms();
//(2)写指令 38H(不检测忙信号)
Write_Cmd_Func(0x38);
//(3)延时 5ms
Delay5ms();
//(4)以后每次写指令,读/写数据操作均需要检测忙信号
//(5)写指令 38H:显示模式设置
Write_Cmd_Func(0x38);
//(6)写指令 08H:显示关闭
Write_Cmd_Func(0x08);
//(7)写指令 01H:显示清屏
Write_Cmd_Func(0x01);
//(8)写指令 06H:显示光标移动设置
Write_Cmd_Func(0x06);
//(9)写指令 0CH:显示开及光标设置}
Write_Cmd_Func(0x0c);
}
void main()
{
// 1.写在哪
char address = 0x80 + 0x04 ;
// 2.写什么
char dt = 'T';
// 初始化操作
LCD1602_INIT();
// 写地址
Write_Cmd_Func(address);
// 写数据
Write_Data_Func(dt);
while(1);
}
写入字符串
// 重复代码已经省略
/*
写入字符串
参数 :
char addr : 写入位置 ,参考上文 ”如何确定位置“
char* dt : 写入数据 ,参考上文 "如何确定内容"
*/
void Write_Str(char addr , char* dt){
char ar = 0x80 + addr;
Write_Cmd_Func(ar);
while(*dt != '\0'){
Write_Data_Func(*dt);
dt++;
}
}
void main()
{
static char* temp = "temp:20 C\0" ;
static char* hum = "hum:50%\0" ;
LCD1602_INIT();
Write_Str(0x00,temp);
Write_Str(0x40,hum);
while(1);
}
今天的分享就到这里了,希望对你有帮助,如有错误的地方敬请批评(当然也可以是意见或建议啦😍😍),本人虚心接受。