從51單片機入門,對流水燈有了一定的認識后,就可以進入學習顯示類的外設驅動,因為學習時往往需要直接驗證、跟蹤結果,顯示類的外設可以把代碼運行情況以非常直觀的方式反饋回來。因此,筆者此處首先分析講解1602字符型液晶的使用。
1. 1602字符型液晶概述1602字符型液晶能夠同時顯示16列2行共32個字符信息。其內部已經存儲了不同的點陣字符圖形,包括阿拉伯數(shù)字、英文大小寫、常用符號等。每個點陣字符圖形都有一個固定的代碼,與我們使用的ASCII碼是一致的。例如大寫英文字母’A’的代碼為0x41,只需在需要顯示的地址位置寫入數(shù)據(jù)0x41即可顯示出字符’A’??捎糜谝恍┖唵涡畔⒔换サ脑O計。
2. 硬件原理圖1602需三根控制線,接單片機P2口第5~7位控制線,采用8位并口接P0口。
3. 驅動編寫我們使用51的IO口來模擬1602的M6800總線,在1602.c中我們實現(xiàn)1602的模塊功能實現(xiàn),內容如下:
#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位數(shù)據(jù)
Delay_5us(); // 數(shù)據(jù)保持時間,約100ns(數(shù)據(jù)手冊)
LCD_EN_SET(); // 拉高EN允許信號
Delay_5us(); // 脈沖保持時間,約450ns(數(shù)據(jù)手冊)
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; // 參數(shù)錯誤,指針為空
}
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(數(shù)據(jù)手冊) 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中實現(xiàn)模塊的接口配置以及一些硬件寄存器的訪問宏實現(xiàn),使之方便移植及修改接口配置。模塊頭文件同時也引出模塊的接口函數(shù),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位數(shù)據(jù) 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來實現(xiàn)調用1602驅動函數(shù),簡單測試調用實現(xiàn)如下: #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); }