自学内容网 自学内容网

【 LCD1602显示屏】使用STC89C51控制1602显示、读写操作时序

LCD1602显示

概述:

LCD1602(Liquid Crystal Display)是一种工业字符型液晶,能够同时显示 16×02 即 32 字符(16列两

在这里插入图片描述

引脚说明

在这里插入图片描述
第 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 脚:背光源负极。

控制指令

在这里插入图片描述

接线

//电源
VSS – GND
VDD – 5V
//对比度
VO – GND
//控制线
RS – P1.0
RW – P1.1
E – P1.4
//背光灯
A – 5V
K – GDN
//数据
D0到D7 – P0.到P0.7

控制思路

在这里插入图片描述

如何显示,在哪里显示?

在这里插入图片描述
例如第二行第一个字符的地址是 40H,那么是否直接写入 40H 就可以将光标定位在第二行第一个字符的位置呢?这样不行,因为写入显示地址时要求最高位 D7 恒定为高电平 1 所以实
际写入的数据应该是01000000B(40H)+10000000B(80H)=11000000B(C0H)

步骤

  • 初始化LCD1602:发送初始化指令,使LCD1602进入8位模式,并设置显示参数(如光标显示、显示开关等)。
  • 指令发送:使用80C51向LCD1602发送命令和数据。通过控制RS、RW和E引脚,并发送8位数据。
  • 编写显示函数:编写函数以在LCD1602上显示字符和字符串,包括光标移动、清屏等功能。

代码示例

#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 EN = P1^4;
void check_busy()
{
char tmp = 0x80;
databuffer = 0x80;
while(tmp & 0x80){//1000 0000
RS = 0;
RW = 1;
EN = 0;
_nop_();
EN = 1;
_nop_();
_nop_();
tmp = databuffer;
EN = 0;
_nop_();
}
}
void Write_Cmd_Func(char cmd)
{
check_busy();
RS = 0;
RW = 0;
EN = 0;
_nop_();
databuffer = cmd;
_nop_();
EN = 1;
_nop_();
_nop_();
EN = 0;
_nop_();    
}
void Write_Data_Func(char dataShow)
{
check_busy();
RS = 1;
RW = 0;
EN = 0;
_nop_();
databuffer = dataShow;
_nop_();
EN = 1;
_nop_();
_nop_();
EN = 0;
_nop_();    
}
void Delay15ms()        
{
unsigned char i, j;
i = 27;
j = 226;
do
{
while (--j);
} while (--i);
}
void Delay5ms()     
{
unsigned char i, j;
//@11.0592MHz
//@11.0592MHz
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 LCD1602_showLine(char row, char col, char *string)
   
{
    
    switch(row){
        case 1:
                Write_Cmd_Func(0x80+col);
                while(*string){
                    Write_Data_Func(*string);
                    string++;
                }
                break;
        
        case 2:
                Write_Cmd_Func(0x80+0x40+col);
                while(*string){
                    Write_Data_Func(*string);
                    string++;
                }
                break;
    
    }
 }
void main()
{
char position = 0x80 + 0x05;
//char dataShow = 'C';
LCD1602_INIT();
LCD1602_showLine(1,5,"NO.1");
LCD1602_showLine(2,0,"hello word");
}

总结

对databuffer = dataShow;的理解

databuffer对应P0口 ,dataShow对应要传输的字符,最终会被编译为ASCII 码。
假设 P0 = 0x68,这意味着你希望将 0x68(即二进制的 01101000)赋值给 P0 端口,使得 P0 端口的 8 个引脚分别对应这个二进制值的每一位。
步骤:

  • 定义P0端口:在51单片机中,P0端口被定义为一个特殊功能寄存器,地址为 0x80。编译器会处理对 P0 的直接赋值
  • 直接赋值:将数据 0x68 赋值给 P0 端口寄存器。此时,P0 端口的8个引脚(P0.0 - P0.7)会被设置为对应的二进制位值

假设调用 LCD1602_showLine(1, 5, “hello”):

  1. 设置光标到第一行的第5列。
  2. 逐个字符发送

第一个字符 h 的 ASCII 码是 0x68,对应的二进制是 01101000。通过 databuffer = ‘h’;,P0 端口的8个引脚将变为 01101000,然后触发使能信号使LCD读取该字符。
第二个字符 e 的 ASCII 码是 0x65,对应的二进制是 01100101。通过 databuffer = ‘e’;,P0 端口的8个引脚将变为 01100101,然后触发使能信号使LCD读取该字符。
依此类推,直到发送完字符串中的所有字符


原文地址:https://blog.csdn.net/qq_40064717/article/details/140463196

免责声明:本站文章内容转载自网络资源,如本站内容侵犯了原著者的合法权益,可联系本站删除。更多内容请关注自学内容网(zxcms.com)!