自制单片机之四……LCD1602的驱动

时间:2022-05-15 13:56:40

  LCD1602已很普遍了,具体介绍我就不多说了,市面上字符液晶绝大多数是基于HD44780液晶芯片的,控制原理是完全相同的,因此HD44780写的控制程序可以很方便地应用于市面上大部分的字符型液晶。字符型LCD通常有14条引脚线或16条引脚线的LCD,多出来的2条线是背光电源线VCC(15脚)和地线GND(16脚),其控制原理与14脚的LCD完全一样,定义如下表所示:

字符型LCD的引脚定义

自制单片机之四……LCD1602的驱动

HD44780内置了DDRAM、CGROM和CGRAM。

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

自制单片机之四……LCD1602的驱动

也就是说想要在LCD1602屏幕的第一行第一列显示一个"A"字,就要向DDRAM的00H地址写入“A”字的代码就行了。但具体的写入是要按LCD模块的指令格式来进行的,后面我会说到的。那么一行可有40个地址呀?是的,在1602中我们就用前16个就行了。第二行也一样用前16个地址。对应如下:
                         DDRAM地址与显示位置的对应关系

自制单片机之四……LCD1602的驱动

我们知道文本文件中每一个字符都是用一个字节的代码记录的。一个汉字是用两个字节的代码记录。在PC上我们只要打开文本文件就能在屏幕上看到对应的字符是因为在操作系统里和BIOS里都固化有字符字模。什么是字模?就代表了是在点阵屏幕上点亮和熄灭的信息数据。例如“A”

字的字模:
           01110     ○■■■○
           10001     ■○○○■
           10001     ■○○○■
           10001     ■○○○■
           11111     ■■■■■
           10001     ■○○○■
           10001     ■○○○■
上图左边的数据就是字模数据,右边就是将左边数据用“○”代表0,用“■”代表1。看出是个“A”字了吗?在文本文件中“A”字的代码是41H,PC收到41H的代码后就去字模文件中将代表A字的这一组数据送到显卡去点亮屏幕上相应的点,你就看到“A”这个字了。
刚才我说了想要在LCD1602屏幕的第一行第一列显示一个"A"字,就要向DDRAM的00H地址写入“A”字的代码41H就行了,可41H这一个字节的代码如何才能让LCD模块在屏幕的阵点上显示“A”字呢?同样,在LCD模块上也固化了字模存储器,这就是CGROM和CGRAM。HD44780内置了192个常用字符的字模,存于字符产生器CGROM(Character Generator ROM)中,另外还有8个允许用户自定义的字符产生RAM,称为CGRAM(Character Generator RAM)。下图说明了CGROM和CGRAM与字符的对应关系。

自制单片机之四……LCD1602的驱动

从上图可以看出,“A”字的对应上面高位代码为0100,对应左边低位代码为0001,合起来就是01000001,也就是41H。可见它的代码与我们PC中的字符代码是基本一致的。因此我们在向DDRAM写C51字符代码程序时甚至可以直接用P1='A'这样的方法。PC在编译时就把“A”先转为41H代码了。
字符代码0x00~0x0F为用户自定义的字符图形RAM(对于5X8点阵的字符,可以存放8组,5X10点阵的字符,存放4组),就是CGRAM了。后面我会详细说的。
0x20~0x7F为标准的ASCII码,0xA0~0xFF为日文字符和希腊文字符,其余字符码(0x10~0x1F及0x80~0x9F)没有定义。

那么如何对DDRAM的内容和地址进行具体操作呢,下面先说说HD44780的指令集及其设置说明,请浏览该指令集,并找出对DDRAM的内容和地址进行操作的指令。

共11条指令:

1.清屏指令

自制单片机之四……LCD1602的驱动

功能:<1> 清除液晶显示器,即将DDRAM的内容全部填入"空白"的ASCII码20H;
               <2> 光标归位,即将光标撤回液晶显示屏的左上方;
               <3> 将地址计数器(AC)的值设为0。

2.光标归位指令

自制单片机之四……LCD1602的驱动

功能:<1> 把光标撤回到显示器的左上方;
                <2> 把地址计数器(AC)的值设置为0;
                <3> 保持DDRAM的内容不变。

3.进入模式设置指令

自制单片机之四……LCD1602的驱动

功能:设定每次定入1位数据后光标的移位方向,并且设定每次写入的一个字符是否移动。参数设定的
                情况如下所示:
                位名              设置
                I/D               0=写入新数据后光标左移                  1=写入新数据后光标右移
                S                 0=写入新数据后显示屏不移动              1=写入新数据后显示屏整体右移1个字符

4.显示开关控制指令

自制单片机之四……LCD1602的驱动

功能:控制显示器开/关、光标显示/关闭以及光标是否闪烁。参数设定的情况如下:
                位名              设置
                 D                0=显示功能关               1=显示功能开
                 C                0=无光标                   1=有光标
                 B                0=光标闪烁                 1=光标不闪烁

5.设定显示屏或光标移动方向指令

自制单片机之四……LCD1602的驱动

功能:使光标移位或使整个显示屏幕移位。参数设定的情况如下:
                S/C               R/L                设定情况
                0                 0                  光标左移1格,且AC值减1
                0                 1                  光标右移1格,且AC值加1
                1                 0                  显示器上字符全部左移一格,但光标不动
                1                 1                  显示器上字符全部右移一格,但光标不动

6.功能设定指令

自制单片机之四……LCD1602的驱动

功能:设定数据总线位数、显示的行数及字型。参数设定的情况如下:
                位名              设置
                DL                0=数据总线为4位                  1=数据总线为8位
                N                 0=显示1行                        1=显示2行
                F                 0=5×7点阵/每字符                1=5×10点阵/每字符

7.设定CGRAM地址指令

自制单片机之四……LCD1602的驱动

功能:设定下一个要存入数据的CGRAM的地址。

8.设定DDRAM地址指令
自制单片机之四……LCD1602的驱动

功能:设定下一个要存入数据的CGRAM的地址。

9.读取忙信号或AC地址指令

自制单片机之四……LCD1602的驱动

功能:<1> 读取忙碌信号BF的内容,BF=1表示液晶显示器忙,暂时无法接收单片机送来的数据或指令;
                    当BF=0时,液晶显示器可以接收单片机送来的数据或指令;
                <2> 读取地址计数器(AC)的内容。

10.数据写入DDRAM或CGRAM指令一览

自制单片机之四……LCD1602的驱动

功能:<1> 将字符码写入DDRAM,以使液晶显示屏显示出相对应的字符;
                <2> 将使用者自己设计的图形存入CGRAM。

11.从CGRAM或DDRAM读出数据的指令一览

自制单片机之四……LCD1602的驱动

功能:读取DDRAM或CGRAM中的内容。

基本操作时序:

读状态           输入:RS=L,RW=H,E=H                                      输出:DB0~DB7=状态字
写指令           输入:RS=L,RW=L,E=下降沿脉冲,DB0~DB7=指令码            输出:无
读数据           输入:RS=H,RW=H,E=H                                      输出:DB0~DB7=数据
写数据           输入:RS=H,RW=L,E=下降沿脉冲,DB0~DB7=数据              输出:无

看了那么多是不是有些晕?我也是啊,不过慢慢理解还是没问题的。
实际上面说了那么多具体怎么操作我还是没会啊?好!咱就简单点。
举个实例,就在LCD1602屏幕上第一行第一列显示个“A”字。
1.先初始化。(老大!好像上面没初始化这条指令啊!)
           先别拿东西扔我,说明书上是这么说的。也就先写入些指令。
//先定义接口
# include <AT89x51.h>
/*****************************************
         P1------DB0~DB7           P2.0------RS
                              P2.1------RW
                              P2.2------E
*****************************************/
# define LCD_DB        P1
        sbit         LCD_RS=P2^0;
        sbit         LCD_RW=P2^1;
        sbit         LCD_E=P2^2;

/******定义函数****************/

# define uchar unsigned char
# define uint unsigned int
void LCD_init(void);//初始化函数
void LCD_write_command(uchar command);//写指令函数
void LCD_write_data(uchar dat);//写数据函数
void LCD_disp_char(uchar x,uchar y,uchar dat);//在某个屏幕位置上显示一个字符,X(0-16),y(1-2)
//void LCD_check_busy(void);//检查忙函数。我没用到此函数,因为通过率极低。
void delay_n40us(uint n);//延时函数
//********************************
//*******初始化函数***************
void LCD_init(void)
{
LCD_write_command(0x38);//设置8位格式,2行,5x7
LCD_write_command(0x0c);//整体显示,关光标,不闪烁
LCD_write_command(0x06);//设定输入方式,增量不移位
LCD_write_command(0x01);//清除屏幕显示
delay_n40us(100);//实践证明,我的LCD1602上,用for循环200次就能可靠完成清屏指令。
}
//********************************
//********写指令函数************
void LCD_write_command(uchar dat)
{
LCD_DB=dat;
LCD_RS=0;//指令
LCD_RW=0;//写入
LCD_E=1;//允许
LCD_E=0;
delay_n40us(1);//实践证明,我的LCD1602上,用for循环1次就能完成普通写指令。
}
//*******************************
//********写数据函数*************
void LCD_write_data(uchar dat)
{
LCD_DB=dat;
LCD_RS=1;//数据
LCD_RW=0;//写入
LCD_E=1;//允许
LCD_E=0;
delay_n40us(1);
}
//********************************
//*******显示一个字符函数*********
void LCD_disp_char(uchar x,uchar y,uchar dat)
{
uchar address;
if(y==1)
         address=0x80+x;
else
         address=0xc0+x;
LCD_write_command(address);
LCD_write_data(dat);
}
//********************************
/*******检查忙函数*************
void LCD_check_busy()      //实践证明,在我的LCD1602上,检查忙指令通过率极低,以
{                                          //至于不能正常使用LCD。因此我没有再用检查忙函数。而使
do                                       //用了延时的方法,延时还是非常好用的。我试了一下,用
        { LCD_E=0;                   //for循环作延时,普通指令只要1次循就可完成。清屏指令
          LCD_RS=0;                 //要用200次循环便能完成。    
          LCD_RW=1;
          LCD_DB=0xff;
          LCD_E=1;
        }while(LCD_DB^7==1);


******************************/
//********延时函数***************
void delay_n40us(uint n)
{ uint i;
      uchar j;            
        for(i=n;i>0;i--)
           for(j=0;j<2;j++);          //在这个延时循环函数中我只做了2次循环,
}                                         //实践证明我的LCD1602上普通的指令只需1次循环就能可靠完成。
//*******************************

//*********主函数*****************
void main(void)
{
        LCD_init();
        LCD_disp_char(0,1,"A");
        while(1);
}

//*******************************

具体电路的制作是很简单的,就接了两个电阻,一个是10欧姆的背光限流电阻,另一个是2K的LCD极板电压调节电阻。这两个电阻的阻值怎么定呢?背光比较简单,它就相当于在后面接了几个发光二极管,任何时候你只要在15、16脚串上个100欧的电位器接上电源,调节电位器,觉得亮度合适。此时的阻值便可。LCD液晶极板驱动电压调节电阻的确定就稍微麻烦一点。在各数据线,控制线接好关通上电源的前提下在第3脚(VEE)和地之间接一个10K的电位器。调节电位器。当3脚电压高时为全亮,电压为0时为全暗(液晶全显示为黑块)。你用电位器把屏幕从全暗刚好调到变亮。这时便可调试程序。待屏幕能正确显示后再细调电位器,使对比度合适。这时的阻值便可确定,然后换成等值的固定电阻焊上便可。

我们接着上次的系统板制做:

自制单片机之四……LCD1602的驱动新买的1602LCD,20元,贵不?自制单片机之四……LCD1602的驱动反面:自制单片机之四……LCD1602的驱动组装后: 自制单片机之四……LCD1602的驱动具体电路图: 自制单片机之四……LCD1602的驱动接口说明: 自制单片机之四……LCD1602的驱动运行:自制单片机之四……LCD1602的驱动用户自定义字符的应用:我们从CGROM表上可以看到,在表的最左边是一列可以允许用户自定义的CGRAM,从上往下看着是16个,实际只有8个字节可用。它的字符码是00000000-00000111这8个地址,表的下面还有8个字节,但因为这个CGRAM的字符码规定0-2位为地址,3位无效,4-7全为零。因此CGRAM的字符码只有最后三位能用也就是8个字节了。等效为0000X111,X为无效位,最后三位为000-111共8个。
如果我们要想显示这8个用户自定义的字符,操作方法和显示CGROM的一样,先设置DDRAM位置,再向DDRAM写入字符码,例如“A”就是41H。现在我们要显示CGRAM的第一个自定义字符,就向DDRAM写入00000000B(00H),如果要显示第8个就写入00000111(08H),简单吧!
好!现在我们来看怎么向这八个自定义字符写入字模。有个设置CGRAM地址的指令大家还记得吗?赶快再找出来看看。自制单片机之四……LCD1602的驱动
从这个指令可以看出指令数据的高2位已固定是01,只有后面的6位是地址数据,而这6位中的高3位就表示这八个自定义字符,最后的3位就是字模数据的八个地址了。例如第一个自定义字符的字模地址为01000000-01000111八个地址。我们向这8个字节写入字模数据,让它能显示出“℃”
地址:01000000  数据:00010000       图示:○○○■○○○○
   01000001     00000110               ○○○○○■■○
   01000010     00001001               ○○○○■○○■
   01000011     00001000               ○○○○■○○○
   01000100     00001000               ○○○○■○○○
   01000101     00001001               ○○○○■○○■
   01000110     00000110               ○○○○○■■○
   01000111     00000000               ○○○○○○○○
下面我们写一段程序让这8个自定义字符显示出一个心的图案:
# include <reg51.h>
unsigned char table1[]={0x03,0x07,0x0f,0x1f,0x1f,0x1f,0x1f,0x1f,
                           0x18,0x1E,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,
                           0x07,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,
                           0x10,0x18,0x1c,0x1E,0x1E,0x1E,0x1E,0x1E,
                           0x0f,0x07,0x03,0x01,0x00,0x00,0x00,0x00,
                           0x1f,0x1f,0x1f,0x1f,0x1f,0x0f,0x07,0x01,
                           0x1f,0x1f,0x1f,0x1f,0x1f,0x1c,0x18,0x00,
                           0x1c,0x18,0x10,0x00,0x00,0x00,0x00,0x00};//心图案
unsigned char table[]={0x10,0x06,0x09,0x08,0x08,0x09,0x06,0x00};//字符℃ #define     CLEARSCREEN     LCD_write_command(0x01)/**************定义接口************************/#define     LCDIO      P2
sbit LCD1602_RS=P3^0;    
sbit LCD1602_RW=P3^1;   
sbit LCD1602_EN=P3^2;   /**************定义函数************************/
void LCD_write_command(unsigned char command);//写入指令函数
void LCD_write_dat(unsigned char dat);//写入数据函数
void LCD_set_xy( unsigned char x, unsigned char y );//设置显示位置函数
void LCD_dsp_char( unsigned x,unsigned char y,unsigned char dat);//显示一个字符函数
void LCD_dsp_string(unsigned char X,unsigned char Y,unsigned char *s);//显示字符串函数
void LCD_init(void);//初始化函数
void delay_nms(unsigned int n);//延时函数
/********************************************//************初始化函数****************/
void LCD_init(void)

CLEARSCREEN;//clear screen 
LCD_write_command(0x38);//set 8 bit data transmission mode 
LCD_write_command(0x0c);//open display (enable lcd display)
LCD_write_command(0x80);//set lcd first display address 
CLEARSCREEN;//clear screen
}
/****************************************************//**************写指令函数********************************/  
void LCD_write_command(unsigned char command)
{
      LCDIO=command;
      LCD1602_RS=0;   
      LCD1602_RW=0;
      LCD1602_EN=0;
      LCD1602_EN=1;
      delay_nms(10);
}
/***************************************************/
/****************写数据函数************************/
void LCD_write_dat(unsigned char dat)
{
LCDIO=dat;
LCD1602_RS=1;
LCD1602_RW=0;
LCD1602_EN=0;
delay_nms(1);
LCD1602_EN=1;
}
/****************************************************//***************设置显示位置**************************/
void LCD_set_xy( unsigned char x, unsigned char y )
{
unsigned char address;
if (y == 1) 
     address = 0x80 + x;
else 
        address =0xc0+ x;
LCD_write_command(address); 
}
/***************************************************//****************显示一个字符**********************/
void LCD_dsp_char( unsigned x,unsigned char y,unsigned char dat)
{
LCD_set_xy( x, y ); 
LCD_write_dat(dat);
}
/**********************************************//***************显示字符串函数***************/
void LCD_dsp_string(unsigned char X,unsigned char Y,unsigned char *s)
{
       LCD_set_xy( X, Y ); 
       while (*s)  
       {
         LCD_write_dat(*s);   
         s ++;
       }
}
/***********************************************//********** 延时**********************/
void delay_nms(unsigned int n)      
{
       unsigned int i=0,j=0;
       for (i=n;i>0;i--)
       for (j=0;j<10;j++);  
}
/**************************************//***********主函数**************/
void main(void)
{
unsigned char i,j,k,tmp;
LCD_init();
delay_nms(100);
tmp=0x40;//设置CGRAM地址的格式字
k=0;
for(j=0;j<8;j++)
     {
        for(i=0;i<8;i++)
         {
           LCD_write_command(tmp+i); // 设置自定义字符的 CGRAM 地址  
           delay_nms(2);
           LCD_write_dat(table1[k]); // 向CGRAM写入自定义字符表的数据
           k++;
           delay_nms(2);
         }
        tmp=tmp+8;
      }
     LCD_dsp_string(1,1,"LCD TEST        ");//在第一行第一列显示“LCD TEST”
     LCD_dsp_string(1,2,"SUCCESSFUL      ");//在第二行第一列显示“SUCCESSFUL”
     for (i=0;i<4;i++)
       {
         LCD_dsp_char( 12+i,1,i);//在第一行第12列位置显示心图案的上半部
         delay_nms(1);
       }
     for (i=4;i<8;i++)
       {
         LCD_dsp_char( 12+i-4,2,i);在第二行第12列位置显示心图案的下半部
         delay_nms(1);
       }
     while (1);
}
/********************************************************************/实际效果如图:自制单片机之四……LCD1602的驱动