从51单片机入门,对流水灯有了一定的认识后,就可以进入学习显示类的外设驱动,因为学习时往往需要直接验证、跟踪结果,显示类的外设可以把代码运行情况以非常直观的方式反馈回来。因此,笔者此处首先分析讲解1602字符型液晶的使用。

1. 1602字符型液晶概述

1602字符型液晶能够同时显示16列2行共32个字符信息。其内部已经存储了不同的点阵字符图形,包括阿拉伯数字、英文大小写、常用符号等。每个点阵字符图形都有一个固定的代码,与我们使用的ASCII码是一致的。例如大写英文字母’A’的代码为0x41,只需在需要显示的地址位置写入数据0x41即可显示出字符’A’。可用于一些简单信息交互的设计。

2. 硬件原理图

02_1602字符液晶显示 - 图1


1602需三根控制线,接单片机P2口第5~7位控制线,采用8位并口接P0口。

3. 驱动编写

我们使用51的IO口来模拟1602的M6800总线,在1602.c中我们实现1602的模块功能实现,内容如下:

#include"reg52.h"

#include"1602.h"

#include<intrins.h>



// 延时nCount * 50微秒(12M)

// 对于STC 1T 51单片机,延时nCount*50/12微秒

voidDelay_50us(unsigned int nCount)

{
      while(nCount--) {
           _nop_();_nop_();_nop_();_nop_();_nop_();

           _nop_();_nop_();_nop_();_nop_();_nop_();

           _nop_();_nop_();_nop_();_nop_();_nop_();

           _nop_();_nop_();_nop_();_nop_();_nop_();

           _nop_();_nop_();_nop_();_nop_();_nop_();

           _nop_();_nop_();_nop_();_nop_();_nop_();

           _nop_();_nop_();_nop_();_nop_();_nop_();

           _nop_();_nop_();_nop_();_nop_();_nop_();

           _nop_();_nop_();_nop_();_nop_();_nop_();

           _nop_();_nop_();_nop_();_nop_();_nop_();

      }

}



static voidDelay_5us()

{
      _nop_();_nop_();_nop_();_nop_();_nop_();



//   Proteus需加长以下延时才能仿真

//   _nop_();_nop_();_nop_();_nop_();_nop_();

//   _nop_();_nop_();_nop_();_nop_();_nop_();

}



static voidLCD_WriteData(unsigned char Dat)

{   

      LCD_RS_SET(); // 拉高RS

      LCD_DATA(Dat); // 输出8位数据

      Delay_5us(); // 数据保持时间,约100ns(数据手册)

      LCD_EN_SET(); // 拉高EN允许信号

      Delay_5us(); // 脉冲保持时间,约450ns(数据手册)

      LCD_EN_CLEAR();  // 清除EN允许信号

}



static voidLCD_WriteCommand(unsigned char Dat)

{   

      LCD_RS_CLEAR(); // 拉低RS

      LCD_DATA(Dat);

      Delay_5us();

      LCD_EN_SET();

      Delay_5us();

      LCD_EN_CLEAR();

}



unsigned charLCD_DisplayString(unsigned char Address, char *pString)

{
      unsigned char i;

      unsigned char MaxAddress;

      if (pString == (void *)0) {
           return 1; // 参数错误,指针为空

      }

      if (Address >= Line1Addr &&Address < Line1Addr+16) {
           MaxAddress = Line1Addr+16; // 地址在1602第一行

      } else if (Address >= Line2Addr&& Address < Line2Addr+16) {
           MaxAddress = Line2Addr+16; // 地址在1602第二行

      } else {
           return 2; // LCD显示地址错误

      }

      LCD_WriteCommand(Address); // 写入显示地址

      Delay_50us(1); // 命令处理时间约40us

      // 字符串结束或到了显示行的最未地址,结束写显示

      for (i=0; Address+i<MaxAddress; i++) {
           if (pString[i] == 0) {
                break;

           }

           LCD_WriteData(pString[i]);

           Delay_50us(1);

      }

      return 0;

}



void LCD_Init()

{
      LCD_RW_CLEAR();

      LCD_EN_CLEAR();

      // 8位总线,双行显示5x7的点阵字符

      LCD_WriteCommand(0x38);    

      // 每个命令处理时间约40us(数据手册)

      Delay_50us(1);

      // 1602开显示,光标不显示

      LCD_WriteCommand(0x0C);

      Delay_50us(1);

      // 光标右移

      LCD_WriteCommand(0x06);

      Delay_50us(1);

      // 清屏,清屏命令处理时间为1.6ms         

      LCD_WriteCommand(0x01);

      Delay_50us(40);

}

我们在模块头文件1602.h中实现模块的接口配置以及一些硬件寄存器的访问宏实现,使之方便移植及修改接口配置。模块头文件同时也引出模块的接口函数,void LCD_Init()用来初始化1602,unsigned char LCD_DisplayString(unsigned char Address, char*pString)用来在指定位置显示字符串信息。其内容如下:

#ifndef __1602_H__

#define __1602_H__



#ifdef __cplusplus

extern "C" {
#endif



sbit LCD_RS = P2^5;

sbit LCD_RW = P2^6;

sbit LCD_EN = P2^7;

#define Line1Addr       0x80 // 1602第一行显示的首地址

#define Line2Addr       0xc0 // 1602第二行显示的首地址



#define LCD_EN_SET() {LCD_EN = 1;}

#define LCD_EN_CLEAR()  {LCD_EN = 0;}

#define LCD_RW_SET()     {LCD_RW = 1;}

#define LCD_RW_CLEAR() {LCD_RW = 0;}

#define LCD_RS_SET() {LCD_RS = 1;}

#define LCD_RS_CLEAR()  {LCD_RS = 0;}

#define LCD_DATA(Dat)    {P0 = (Dat);} // P0口输出8位数据



void LCD_Init(void);

unsigned charLCD_DisplayString(unsigned char Address, char *pString);

void Delay_100us(unsigned intnCount);



#ifdef __cplusplus

}

#endif



#endif /*__1602_H__*/

外部模块通过引入1602的模块头文件1602.h来实现调用1602驱动函数,简单测试调用实现如下:

#include "reg52.h"

#include "1602.h"



void main()

{
      //需显示的字符串1

      codechar String1[] = {
           "huang20083200056"

      };

      //需显示的字符串2

      codechar String2[] = {
           "QQ:1048272975"

      };

      //初始化1602

      LCD_Init();

      //在第一行首地址开始显示字符串1

      LCD_DisplayString(Line1Addr,String1);

      //在第二行第二个地址开始显示字符串2

      LCD_DisplayString(Line2Addr+1,String2);

      while(1);         

}

附注:
此章节的Keil工程,包含源码,可自行用Proteus仿真验证,1602.rar。