/
程序作者:yuki1406
程序作用:STC89CXX的1602液晶8BIT *** 作模式 测试环境MCU:STC89C51RC 晶振频率110592MHZ 以及外部晶振24MHZ加1倍频到48M测试正常
编写时间:2012年07月17日
结束时间:2012年07月17日
程序版本:V11
使用说名:
先调用初始化
然后检测忙状态
写入地址
检测忙状态
写入数据
while(LCD1602_readway()); 忙状态检测
器件说明:
显示容量:162个字符
芯片工作电压:45-55v
工作电流:2MA(50v)
模块最佳工作电压:5V
字符尺寸:295x435(WXH)mm
接口信号说明
编号 符号 引脚说明 编号 符号 引脚说明
1 VSS 电源地 9 D2 DATA 1/0
2 VDD 电源正极 10 D3 DATA 1/0
3 VL 液晶显示偏压信号 11 D4 DATA 1/0
4 RS 数据/命令选择端(H/L) 12 D5 DATA 1/0
5 R/W 读/写选择端(H/L) 13 D6 DATA 1/0
6 E 使能信号 14 D7 DATA 1/0
7 D0 DATA 1/0 15 BLA 背光源正极
8 D1 DATA 1/0 16 BLK 背光源负极
基本 *** 作时序:
11读状态:输入:RS=L,RE=H,E=H 输出:D0-D7=状态字
12写状态:输入:RS=L,RE=L,D0-D7=指令码,E=高脉冲 输出:无
13读数据:输入:RS=H,RE=H,E=H 输出:D0-D7=数据
14写数据:输入:RS=H,RE=L,D0-D7=数据,E=高脉冲 输出:无
指令说明
初始化设置
11显示模式设置
指令码 功能
00111000 设置16x2显示 5x7点阵 8位数据接口
显示开/关及光标设置
指令码 功能
00001DCB D=1开显示: D=0 关显示
C=1显示光标:C=0 不显示光标
B=1光标显示:B=0光标不显示
000001NS N=1当读或写一个字符后地址指针加一,且光标加一
N=0当读或写一个字符后地址指针剪一,且光标减一
S=1当写一个字符,整平显示左移(N=1)
或右移(N=0),以得到光标不移动而屏幕移动的效果。
S=0当写一个字符,整屏显示不移动
程序大小:108字节
更新时间:
发现每次写入时 都手动调用读忙函数 很麻烦 把读忙函数加到了写入函数内
更新版本:无
更新日志:无
/
#ifndef __LCD_1602_8BIT_H__
#define __LCD_1602_8BIT_H__
#include<stc89h> //包含单片机寄存器的头文件
//#include<intrinsh> //包含NOP空指令的头文件
#define uchar unsigned char
#define uint unsigned int
#define LCD1602_H 1 //宏定义手册中出现的H的定义
#define LCD1602_L 0 //宏定义手册中出现的L的定义
#define LCD1602_DAT 1 //数据
#define LCD1602_COM 0 //命令
#define LCD_15MS 300 //宏定义15MS延时需要的数值
#define LCD_5MS 100 //宏定义 5MS延时需要的数值
/显示模式指令/
#define LCD_Display_mode 0X38 //设置16x2显示 5x7点阵 8位数据接口
/显示开/关及光标设置/
#define LCD_shows0 0X0C //开显示 不显示光标 光标不闪烁
#define LCD_shows2 0X0E //开显示 显示光标 光标不闪烁
#define LCD_shows1 0X0F //开显示 显示光标 光标 闪烁
#define LCD_shows3 0X08 //关显示 不显示光标 光标不闪烁
/指针设置/
#define LCD_cursor1 0X04 //写一个字符 地址指针减1
#define LCD_cursor2 0X05 //写一个字符 地址指针减1 并屏幕右移
#define LCD_cursor3 0X06 //写一个字符 地址指针加1
#define LCD_cursor4 0X07 //写一个字符 地址指针加1 并屏幕左移
/清屏指令/
#define LCD_clear 0x01 //清屏指令 数据指针清零 所有显示清零
/忙状态字/
#define LCD_WAY 0x80 //状态字
/宏定义显示起始地址/
#define LCD_ADDH 0X80 //第一行地址0x80-0xA7
#define LCD_ADDL 0XC0 //第二行地址0xC0-0xE7
/IO口定义/
#define LCD1602_DATA P0 //宏定义8位数据线IO为P0口 D0~D7=P00~P07 8位数据线 D0=P00;
sbit LCD1602_RS=P2^5; //数据/命令选择端(H/L)
sbit LCD1602_RW=P2^6; //读/写选择端(H/L)
sbit LCD1602_E =P2^7; //使能信号
/函数声明/
void LCD1602_init(); //液晶初始化函数
void LCD1602_writecd(bit lcd_rs, uchar LCD1602_d);//写命令/数据 函数bit lcd_rs是数据还是命令 uchar LCD1602_d要写入的数据
uchar LCD1602_readway(); //读忙状态函数 由写入和读取函数调用
uchar LCD1602_readata(); //读数据函数
void LCD_DELAY(uchar LCD_delay); //
/液晶初始化函数/
void LCD1602_init() //液晶初始化函数
{ LCD_DELAY(LCD_15MS); //延时15MS 初始化
LCD1602_writecd(LCD1602_COM,LCD_Display_mode);//写指令38H 设置16x2显示 5x7点阵 8位数据接口
LCD1602_writecd(LCD1602_COM,LCD_shows0); //开显示 不显示光标 光标不闪烁
LCD1602_writecd(LCD1602_COM,LCD_cursor1); //检查忙状态
LCD1602_writecd(LCD1602_COM,LCD_clear); //写指令01H:显示清屏
}
/液晶写命令/数据函数/
void LCD1602_writecd(bit lcd_rs, uchar LCD1602_cd)//写命令/数据 函数
{ uchar LCD1602_NUM; //定义变量用来液晶无忙回答的退出死循环
LCD1602_NUM=255; //忙状态检测次数
while(LCD1602_readway()) //检查忙状态
{LCD1602_NUM--; //检测次数自减
if(LCD1602_NUM==0) //判断检测次数等于0
{break;} //退出循环判断忙
}
LCD1602_RW = LCD1602_L; //读/写选择端(H/L)
LCD1602_RS = lcd_rs; //数据/命令选择端(H/L)
LCD1602_DATA= LCD1602_cd; //IO口赋值
LCD1602_E = LCD1602_H; //拉高使能信号开始传输数据
LCD1602_E = LCD1602_L; //拉低使能信号锁存数据
LCD1602_DATA= 0xff; //IO口数据清除
}
/忙状态读取函数/
uchar LCD1602_readway()//读状态函数 由写入和读取函数调用
{ uchar LCD1602_way; //状态字变量
LCD1602_DATA=0xff; //IO口数据清除
LCD1602_RS = LCD1602_COM; //数据/命令选择端(H/L) 命令
LCD1602_RW = LCD1602_H; //读/写选择端(H/L)
LCD1602_E = LCD1602_H; //拉高使能信号开始接收状态
LCD1602_way = LCD1602_DATA; //读取状态
LCD1602_E = LCD1602_L; //拉低使能信号锁存数据
LCD1602_way = LCD1602_way&LCD_WAY;//取忙状态字
return(LCD1602_way); //返回状态字
}
/液晶读数据函数/
uchar LCD1602_readata()//读数据函数
{ uchar LCD1602_data; //数据暂存变量
while(LCD1602_readway()); //检查忙状态 --------------------
LCD1602_DATA=0xff; //IO口数据清除
LCD1602_RS = LCD1602_DAT; //数据/命令选择端(H/L) 数据
LCD1602_RW = LCD1602_H; //读/写选择端(H/L)
LCD1602_E = LCD1602_H; //拉高使能信号开始接收状态
LCD1602_data= LCD1602_DATA; //读取状态
LCD1602_E = LCD1602_L; //拉低使能信号锁存数据
return(LCD1602_data); //返回数据
}
/延时函数/
void LCD_DELAY(uchar LCD_delay)//
{ uchar lcd_del;
while(LCD_delay--) //自减
{lcd_del=100;
while(lcd_del--);}
}
我写的 参考下吧 只写了驱动 此程序不会显示任何东西 调用初始化 会看到显示屏 清屏的效果
这是我写的一个成功的,你仔细对比下就知道了!
#include<reg52h>
#include<intrinsh>
#define U8 unsigned char
#define U16 unsigned int
#define BOOL bit
U8 secondh,secondl,minuteh,minutel,hourh,hourl,second,minute,hour;
U8 monthl,monthh,month,day,datel,dateh,date,year,yearl,yearh;
/定义LCD功能管脚/
sbit LCDRS = P2^6 ; //控制LCD数据或命令的读写
sbit LCDRW = P2^5 ; //控制LCD读写信号
sbit LCDEN = P2^7 ; //使能LCD
//
/定义DS1302功能管脚/
sbit DSRST = P3^5 ; //DS1302复位管脚
sbit DSIO = P3^4 ; //控制DS1302数据传输
sbit DSCLK = P3^6 ; //DS1302时钟
//
/延时函数/
void delay(U16 time)
{
U16 time1,time2;
for(time1=time;time1>0;time1--)
for(time2=110;time2>0;time2--);
}
//
/检测LCD忙碌状态/
BOOL judge_lcd()
{
BOOL flag;
LCDEN = 0;
LCDRS = 0;
LCDRW = 1;
LCDEN = 1;
delay(1);
flag=(BOOL)(P0&0x80); //状态位第7位为1时禁止读写,为0时可读写
LCDEN = 0;
return flag;
}
//
/实现LCD写指令/
void write_lcd_command(U16 command)
{
while(judge_lcd()); //检测LCD是否为忙碌状态
P0 = command ;
LCDRS = 0 ;
LCDRW = 0 ;
LCDEN = 1 ;
delay(1);
LCDEN = 0 ;
}
//
/实现LCD写数据/
void write_lcd_date(U16 date)
{
while(judge_lcd()); //检测LCD是否为忙碌状态
P0 = date ;
LCDRS = 1 ;
LCDRW = 0 ;
LCDEN = 1 ;
delay(1);
LCDEN = 0 ;
}
//
/实现LCD写指令及数据/
void write_lcd_com_date(U16 addr,U16 date)
{
write_lcd_command(addr);
delay(1);
write_lcd_date(date);
}
//
/初始化LCD/
void init_lcd()
{
write_lcd_command(0x38); //设置16x2显示,5x7点阵,8位数据口
delay(1);
write_lcd_command(0x0c); //开显示,不显示光标,光标不闪烁
delay(1);
write_lcd_command(0x06); //读写一个字符后,地址自动加1,不移动屏幕
delay(1);
write_lcd_command(0x01); //清屏
delay(1);
}
//
/读DS1302/
U8 read_ds1302(U8 addr)
{
U8 i,temp=0x00;
DSRST=0;
DSIO=1;
DSCLK=0;
DSRST=1;
for (i=0;i<8;i++) //循环8次 写入地址数据
{
DSCLK=0;
DSIO=addr&0x01; //每次传输低字节
addr>>=1; //右移一位
DSCLK=1;
}
delay(1);
if(DSIO)
temp|=0x80; //每次传输低字节
DSCLK=0;
temp>>=1;
for (i=0;i<7;i++) //循环8次 读取数据
{
DSCLK=0;
if(DSIO)
temp|=0x80; //每次传输低字节
DSCLK=1;
temp>>=1; //右移一位
}
DSCLK=1;
DSRST=0;
DSIO=0;
return temp; //返回
}
//
/写DS1302/
void write_ds1302(U8 addr, U8 date)
{
U8 i;
DSRST=0;
DSCLK=0;
DSRST=1;
for (i=0;i<8;i++) //循环8次 写入地址数据
{
DSCLK=0;
_nop_();
DSIO=addr&0x01; //每次传输低字节
addr>>=1; //右移一位
DSCLK=1;
_nop_();
}
for (i=0;i<8;i++) //循环8次 写入数据
{
DSCLK=0;
_nop_();
DSIO=date&0x01; //每次传输低字节
date>>=1; //右移一位 DSCLK=1
DSCLK=1;
_nop_();
}
DSRST=0;
delay(1);
}
//
/初始化DS1302/
void init_ds1302()
{
//读秒
if((((second=read_ds1302(0x81))>>7)&&0x01)==1)
{
write_ds1302(0x8e,0x00); //关闭写保护
write_ds1302(0x80,0x00); //00秒
write_ds1302(0x82,0x54); //54分
write_ds1302(0x84,0x17); //17点
write_ds1302(0x86,0x14); //14日
write_ds1302(0x88,0x10); //10月
write_ds1302(0x8a,0x05); //星期五
write_ds1302(0x8c,0x11); //2011年
write_ds1302(0x8e,0x80); //允许写保护
}
}
//
/主函数/
void main()
{
init_lcd();
delay(2);
write_lcd_com_date(0x81,'2');
write_lcd_com_date(0x82,'0');
delay(1);
init_ds1302();
delay(1);
while(1)
{
second=read_ds1302(0x81);
secondl=second&0x0f;
secondh=second>>4;
minute=read_ds1302(0x83);
minutel=minute&0x0f;
minuteh=minute>>4;
hour=read_ds1302(0x85);
hourl=hour&0x0f;
hourh=hour>>4;
month=read_ds1302(0x89);
monthl=month&0x0f;
monthh=month>>4;
year=read_ds1302(0x8d);
yearl=year&0x0f;
yearh=year>>4;
date=read_ds1302(0x87);
datel=date&0x0f;
dateh=date>>4;
day=read_ds1302(0x8b);
day=day&0x0f;
delay(15);
write_lcd_com_date(0xca,0x30+secondl);
write_lcd_com_date(0xc9,0x30+secondh);
write_lcd_com_date(0xc8,':');
write_lcd_com_date(0xc7,0x30+minutel);
write_lcd_com_date(0xc6,0x30+minuteh);
write_lcd_com_date(0xc5,':');
write_lcd_com_date(0xc4,0x30+hourl);
write_lcd_com_date(0xc3,0x30+hourh);
write_lcd_com_date(0x84,0x30+yearl);
write_lcd_com_date(0x83,0x30+yearh);
write_lcd_com_date(0x85,'-');
write_lcd_com_date(0x87,0x30+monthl);
write_lcd_com_date(0x86,0x30+monthh);
write_lcd_com_date(0x88,'-');
write_lcd_com_date(0x8a,0x30+datel);
write_lcd_com_date(0x89,0x30+dateh);
write_lcd_com_date(0x8c,0x30+day);
}
}
51单片机对lcd1602一些基础程序
#include <intrinsh>
#define dataport P1
sbit RS=P2^ 6;
sbit RW=P2^5;
sbit EN=P2^4;
//========================
//=========================
void waitfor() //检测忙信号函数
{
dataport=0xff;
RS=0;RW=1;_nop_(); //选择指令寄存器 读 *** 作
EN=1;_nop_(); //使能 *** 作
while(dataport&0x80); //如果最高位是1 表示1602正忙 原地踏步 忙完后芯片会将高位拉低
EN=0;
}
//======================
void writedata(unsigned char dataw) //写数据到lcm
{
waitfor(); //测忙
RS=1;RW=0;_nop_(); //选择数据寄存器 写 *** 作
dataport=dataw;_nop_(); //将数据送到数据口
EN=1;_nop_();_nop_();EN=0; //使能
}
//==========================
void writecmd(unsigned char cmd) //写命令到lcm
{
waitfor();
RS=0;RW=0;_nop_();
dataport=cmd;_nop_();
EN=1;_nop_();_nop_();EN=0;
}
//===========================
void init(void) // 初始化函数
{
writecmd(0x38); //功能设定 8位数据传输 双行显示
writecmd(0x0c);//显示器开关
writecmd(0x01);//清屏
writecmd(0x06);//字符进入模式 每进入一个字符光标向右移动一格 原有字符不动
//我在刚开始学的时候不知道下一个字符显示在哪 是和AC值有关还是和光标位置有关
//最后摸索出来是只和光标定位有关 现在还是不知道Ac值有什么用
}
//=========================
void location(unsigned char x,unsigned char y) //确实坐标函数
{
unsigned char temp;
temp=x&0x0f; //只要x数据的后四位
if(y){temp=temp|0x40;} //第一行为0 第二行为1 如果y=1则地址加0x40
temp|=0x80; //DDRAM地址的命令DB7为一
writecmd(temp);
}
//==============================
void displyonechar(unsigned char x,unsigned char y,unsigned char dataw) //显示一个字符函数
{
location(x,y);
writedata(dataw);
}
//=======================================
void displylistchar(unsigned char x,unsigned char y,unsigned char p) //显示字符串
{
while(p) //当一个字符型数组读完时P指的为零
{
displyonechar(x,y,(p++));
x++;
}
}
//=====================================================
void writecgram(unsigned char address,unsigned char p) //写CGRAM的数据
{
unsigned char i=8;
writecmd(address); //CGRAM里的地址 初始值0x40 每次加0x80
while(i--)
{
writedata(p);
p++;
}
}
//=====================================================
void displyonecharacter(unsigned char x,unsigned char y,unsigned char address,unsigned char p) //显示一个自定义字符
{
unsigned char i=8;
writecmd(address); //CGRAM里的地址 初始值0x40 每次加0x08
while(i--)
{
writedata(p);
p++;
}
//============================================================
location(x,y); //设定要显示的位置
writedata((address&=0x3f)/0x08); //要从CGRAM中读出数据在1602上显示 搞了半天发现CGRAM里的地址
} //和DDRAM里的地址有上面的转换关系
//========================================================
void displynumber(unsigned char x,unsigned char y,unsigned long num) //显示一个整数
{
unsigned int number[8];
int k,gh;
for(k=0;;k++)
{
(number+k)=(unsigned int)(num%10);//强制类型转换
num=num/10;
if(num==0)break;
}
for(gh=k;gh>=0;gh--)
{
displyonechar(x,y,((number+gh)+48));
x++;
}
}
//字型码
uchar code nin[]={0x08,0x0f,0x12,0x0f,0,0x1f,0x02,0x02};// "年"
uchar code yue[]={0x0f,0x09,0x0f,0x09,0x0f,0x09,0x0b,0x11};// "月"
uchar code ri[]={0x1f,0x11,0x11,0x1f,0x11,0x11,0x11,0x1f};// "日"
显示汉字
displyonecharacter(0,0,0x40,nin);
displyonecharacter(1,0,0x80,yue);
displyonecharacter(1,0,0xc0,ri);
以上就是关于我在学单片机但是不会打开1602的液晶,求大神帮忙编下程序,资料如下,不够再发,最好是C语音的,全部的内容,包括:我在学单片机但是不会打开1602的液晶,求大神帮忙编下程序,资料如下,不够再发,最好是C语音的,、单片机1602液晶显示的DS1302实时时钟帮我分析以下程序运行步骤,详细解释一下,尤其是Bcd转换、给个单片机液晶1602显示汉字的程序,谢谢等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)