51单片机开发系列之1602字符液晶显示

2019-10-12来源: eefocus关键字:51单片机  1602字符  液晶显示

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


1. 1602字符型液晶概述

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


2. 硬件原理图

 

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


3. 驱动编写


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


#include"reg52.h"

#include"1602.h"

#include

// 延时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

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[] = {

};

//初始化1602

LCD_Init();

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

LCD_DisplayString(Line1Addr,String1);

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

LCD_DisplayString(Line2Addr+1,String2);

while(1);

}


关键字:51单片机  1602字符  液晶显示 编辑:什么鱼 引用地址:http://news.eeworld.com.cn/mcu/ic476933.html 本网站转载的所有的文章、图片、音频视频文件等资料的版权归版权所有人所有,本站采用的非本站原创文章及图片等内容无法一一联系确认版权者。如果本网所选内容的文章作者及编辑认为其作品不宜公开自由传播,或不应无偿使用,请及时通过电子邮件或电话通知我们,以迅速采取适当措施,避免给双方造成不必要的经济损失。

上一篇:课程设计题一:八人抢答器
下一篇:51单片机----16*16LED点阵

关注eeworld公众号 快捷获取更多信息
关注eeworld公众号
快捷获取更多信息
关注eeworld服务号 享受更多官方福利
关注eeworld服务号
享受更多官方福利

推荐阅读

STM32F030F4P6单片机由外部时钟改为内部时钟的步骤
; else  { /* If HSE fails to start-up, the application will have wrong clock          configuration. User can add here some code to deal with this error */  }  } 三.接下来我们想将系统时钟改为内部时钟,修改办法依旧是修改  SetSysClock()这个函数中的配置.首先我们要知道单片机内部的时钟到底是怎样一种
发表于 2019-10-09
STM32F030F4P6单片机由外部时钟改为内部时钟的步骤
msp430单片机通过串口向上位机发送浮点数
组装数据 首先将数据组装起来,通过sprintf函数。 组装数据时用的小技巧:    length = 0;     sprintf(Storage+length, "Rr=%4.3f ", g_StrComm.aSBuf[0]);    length = strlen(Storage);   //strlen =12    sprintf(Storage+length, "Rh=%4.3f ", g_StrComm.aSBuf[2]);   
发表于 2019-10-08
单片机用串口传输浮点型数据方法
我们都知道,单片机串口传输的单位是字节,而浮点型数是占四个字节,简单思路是用一个char型指针指向浮点型数据,利用指针寻址即可以将浮点数拆成四个char数据。接收端接收到四个char型数,为了还原成float型数据,采用共用体是一不错的方式。测试代码如下:      程序运行结果如下:       注意:结构体中如果用有符号字符型数据,在printf打印时有出点小问题,大家可以试试看的,嘿嘿当然还有个嵌入式常说的大小端问题,相信大家都明白咋咋回事了。
发表于 2019-10-08
单片机用串口传输浮点型数据方法
课程设计题三十三:基于单片机的电子钟
(150); P0 = P1 = P2 = P3 = 0xff; //单片机IO口初始化为1 init_time0(); //初始化定时器 init_1602(); //lcd1602初始化 init_1602_dis_csf(); //lcd1602初始化显示 init_eeprom();       //开始初始化保存的数据 delay_1ms(650); while(1) { key(); //按键程序 if(key_can < 10) { if(flag_beep_en == 0)  //只有闹钟关了的时候才能进入设置 key_with
发表于 2019-09-29
课程设计题三十三:基于单片机的电子钟
课程设计题十四:双机通信
要求:        独立程序的串行接收/发送,两个单片机利用串行口进行方式1的全双工串行通信,两个单片机分别带有键盘和八段码数码管,当A单片机按下键盘上的数字时,该数字传送到B单片机上并显示在B单片机上连接的数码管上,反之亦然。可选的波特率为:1200,2400,4800,9600。一、Protues仿真图:二、程序源码:因为注释非常的全,这里就不再进行讲解了。发送端
发表于 2019-09-29
课程设计题十四:双机通信
51单片机pwm信号模拟
适用所有单片机,晶振为11.0592M#include "reg52.h"  int pwm1=1;    第一路pwm占空比int pwm2=60;   第二路pwm占空比sbit led1=P1^0;    sbit led2=P1^3;sbit md1=P1^0;void pwm_init(){TMOD|=0x01;TH0=(65536-100)/256;       约1Kmz      TL0=(65536-100)%256;ET0
发表于 2019-09-26
小广播
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号 电信业务审批[2006]字第258号函 京公海网安备110108001534 Copyright © 2005-2019 EEWORLD.com.cn, Inc. All rights reserved