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);
void delay_1ms(uint x)//延时
{ uchar j; //加一变量 uchar i,j;
while((x--)!=0) //改为for(i=0;i<x;i++)
{ for(j=0;j<125;j++) ;
}
}
//检查LCD忙状态
bit lcd_busy() //其实大多数都delayms(5)代替的,尤其是仿真时不能用此函数
{
bit result;
rs=0;
rw=0;
e = 1;
delay_1ms(1);
result = (bit)(data&0x80); //不知data是不是已经#define了
e = 0;
return(result);
}
//写指令数据到LCD
void write1602_com(uchar com)
{while(lcd_busy());
rs = 0;
rw = 0;
e = 0;
delay_1ms(1);
data= com;
delay_1ms(1);
e = 1;
delay_1ms(1);
e = 0;
}
//写显示数据到LCD
void write1602_data(uchar dat)
{while(lcd_busy());
rs = 1;
rw = 0;
e = 0;
data = dat;
delay_1ms(1);
e = 1;
delay_1ms(1);
e = 0;
}
void start(void) //LCD初始化
{delay_1ms(15);
write1602_com(0x38);
delay_1ms(5);
write1602_com(0x38);
delay_1ms(5);
write1602_com(0x38);
delay_1ms(5);
write1602_com(0x0F);//显示开,关光标
delay_1ms(5);
write1602_com(0x06);//移动光标
delay_1ms(5);
write1602_com(0x01);//清除LCD的显示内容
delay_1ms(5);
}
/ 设定显示位置 /
/
void lcd_data(uchar dat) //1602两行地址不是连续的,第二行是0xC0开始的
{
write1602_com(dat|0x80); //数据指针=80+地址变量
}
/
void lcd_data(uchar dat)
{
unsigned char p;
if (pos>=0x10)
p=pos+0xb0;
else
p=pos+0x80;
write1602_com(p);
write1602_data(c);
}
void main()
{
start();
delay_1ms(10);
write1602_com(0x01); //已经清屏,不知道这句的意义
delay_1ms(10);
delay_1ms(5);
lcd_data(0x00);
delay_1ms(5);
for(i=0;i<12;i++)
{
if(i==13){ //建议这句话加上大括号,放到后边,放在这影响不好
while(1);}
write1602_data(table2[i]);//应先写地址再写数据,而你这地址是不变的,最终导致数组数据在
//一个地址上陆续显示,不知道会是啥效果,并且希望你的数组不是const的。
//可以写个连续显示的函数,加指针的
delay_1ms(1);
}
}
我这1602的程序及仿真还有很多,可以向我联系的
LCD1602忙信号是在D7位,那最高位上,在单片机这个引脚上是否还接有其它什么元件?
还有查询忙状态的程序代码是否对呀?
是仿真,还是实物,要是仿真的话,可以不用查询忙状态的,可以简化程序的。毕竟仿真不同于实物的。
要是实物,1602的屏是否是坏了。你得从各个方面找原因哪。
说实话你的代码相当混乱,并且没有读通LCD1206 Datasheet,给一个我多年前学习时候写的程序给你参考:
#include "STC12C5A60S2H"
#include "intrinsH"
#define uchar unsigned char
#define uint unsigned int
#define FOSC 11059200
#define RS P2_5
#define RW P2_6
#define E P2_7
#define LcdData P0
/函数声明/
//void Check_Busy();
void WriteCommand(uchar command);
void WriteData(uchar dat);
void Iinit_Lcd();
void Show_char(uchar a,uchar str);
void Delay(uint x);
void Show_Vdata(uchar ptr);
/自定义函数/
/延时函数/
void Delay_S(uint x)
{
uint ii;
while(x>0)
{
for(ii=6000;ii>0;ii--);
x--;
}
}
//延时函数
void Delay(uint x)
{ uint i;
uchar j;
_nop_();
for(i=x;i>0;i--)
for(j=0;j<2;j++); //在这个延时循环函数中我只做了
}
/初始化LCD/
void Iinit_Lcd()
{
/
WriteCommand(0x38);//设置8位格式,2行,5x7
WriteCommand(0x0c);//显示命令4,整体显示,关光标,不闪烁
//WriteCommand(0xff);//整体显示,关光标开,闪烁
//WriteCommand(0x04);//光标左移,屏幕文字不移动
//WriteCommand(0x1C); //显示命令5,11000表示文字移动,左移。
WriteCommand(0x06);//显示命3,110光标左移,屏幕文字不移动。
//WriteCommand(0x02);//光标返回00H
WriteCommand(0x01);//清显示
Delay(200);//延时时间短后会出现前面字丢掉的情况。
/
//指今1功能“清显示”00为RS&R/W的电平00000001
//WriteCommand(0x01);//表示清屏。
//指令2功能“光标复位,返回00H地址”00为RS&R/W的电平 0000001其中“”表示任意状态,第二位1表示指令2的意思。
//WriteCommand(0x02);//表示光标复位。
//指令3功能“置输入模式”00为RS&R/W的电平 000001I/DS其中“I/D”为1表示光标右移,为0表示左移,“S”为1表示屏幕文字是否左移,为0表示无效。
//WriteCommand(0x07); //表示光标右移,屏幕整体右移一位。实验为全部字左移。
//WriteCommand(0x06); //表示光标右移,文字不移动。
WriteCommand(0x05); //示光标左移,屏幕整体左移。
//WriteCommand(0x04); //表示光标右移,文字不移动。
//指令4功能“显示开/关控制”00为RS&R/W的电平0001DCB其中“D”表示控制整体的显示开&关1为显示,0为关显示,“C”表示控制光标的开&关,1
//为有光标,0为无光标,“B”控制光标是否闪烁,1为闪烁,0为不闪烁。
//WriteCommand(0x08);//表示关显示,无光标,不闪烁。
//WriteCommand(0x09);//表示关显示,无光标,闪烁。
//WriteCommand(0x0A);//表示关显示,有光标,不闪烁。
//WriteCommand(0x0B);//表示关显示,有光标,有闪烁。
//WriteCommand(0x0C);//表示开显示,无光标,不闪烁。
//WriteCommand(0x0D);//表示开显示,无光标,有闪烁。
//WriteCommand(0x0E);//表示有显示,有光标,不闪烁。
WriteCommand(0x0F);//表示有显示,有光标,有闪烁。
//指令5功能“光标或字符移位”00为RS&R/W的电平0001 S/C R/L其中“S/C”为1表示显示移动的文字,为0时表示移动光标,“R/L”表示移动的
//方向,“”表示任意的无效数。
//WriteCommand(0x10);//表示移动光标,向左移动。
///WriteCommand(0x14);//表示移动光标,向右移动。
WriteCommand(0x18);//表示移动文字,向左移动。
//WriteCommand(0x1C);//表示移动文字,向右移动。
//指令6功能“设置功能”00为RS&R/W的电平001 DL N F其中“DL”为1表示为8位总线,为0时表示为4位总线,“N”为0时单行显示,为1时双行显
//示,“F”为0时显示57点阵字符,为1时显示510点阵字符。
//WriteCommand(0x20);//表示4位总线,单行显示,57点阵。
//WriteCommand(0x24);//表示4位总线,单行显示,510点阵。
//WriteCommand(0x28);//表示4位总线,双行显示,57点阵。
//WriteCommand(0x2C);//表示4位总线,双行显示,510点阵。
//WriteCommand(0x30);//表示8位总线,单行显示,57点阵。
//WriteCommand(0x34);//表示8位总线,单行显示,510点阵。
WriteCommand(0x38); //表示8位总线,双行显示,57点阵。
//WriteCommand(0x3C);//表示8位总线,双行显示,510点阵。
//指令7功能“置字符发生存贮器RAM地址设置”00为RS&R/W的电平01表示字符发生器RAM地址设置命令,后8位表示字符发生存储器地址。
//指令8功能“置数据存贮器DDRAM地址设置”001显示数据存贮器地址
//指令9功能“读忙信号和光标地址 BF置”读忙标志或地址01 BF计数器地址
//指令10功能“写数到CGRAM或DDRAM)10为RS&R/W的电平,其它8位为要写的数据内容。
//指令11功能“从CGRAM或DDRAM读数“11为RS&R/W的电平,其它8位为读出的数据内容。
WriteCommand(0x01);//表示清屏。
WriteCommand(0x02);//表示光标复位。
Delay(500);//延时时间短后会出现前面字丢掉的情况。 /
}
/LCD写数据函数/
//RS=1,RW=0,E=0;
void WriteData(uchar dat)
{
E=0;
LcdData=dat;
RS=1;
RW=0;
E=1;//写数据过程。
Delay(10);//时间短了文字显示不出来,LCD反映不过来。
E=0;//下降沿数据写入。
//Check_Busy();
}
/LCD写指今函数/
//RS=0,RW=0,E=0;
void WriteCommand(uchar command)
{
E=0;
RS=0;
RW=0;
LcdData=command;
Delay(10);//原为100
E=1;
Delay(10);//原为100
E=0;
}
/LCD读数据函数/ //RS=1,RW=1,E=0;
/LCD读指今函数/ //RS=0,RW=1,E=0;
/读忙函数/
/void Check_Busy()
{
E=0;
RS=0;
RW=1;
E=1;
//LcdData=0x00;//拉高数据端为1
//LcdData^7==0;
while(LcdData^7==1);//判忙标志是否为1表示忙等
待,为0跳出。
}
/
/显示字符函数/
void Show_char(uchar a,uchar str)//a表示LCD显示行地址,b表示LCD上字符是否滚动显示,str为字串指针。
{
uchar address;
uchari,j,k;
//Iinit_Lcd(); //如果初始化函数在这里将造成主函数循环中显示屏很闪烁
if(a==1)
{
address=0x80;//LCD第一行首地址
WriteCommand(address);
while(str!='\0')
{
WriteData(str);
str=(str++);
Delay_S(40);//延时后才能看到左移的效果。
}
Delay_S(40);
}
/模似左移/
else
{
for(i=16;i>1;i--)
{
WriteCommand(0xC0);//第二行首地址
for(k=0;k<16;k++)
{
WriteData('');//清空显示数据,因为写数据时LCD地址计数器会自动加一,故只需写数据就行了。
}
address=0xBE;
address=0xBE+i; //最后一个地址开始写,不断减一实现左移。
WriteCommand(address);
_nop_();
j=0;
while((str+j)!='\0')//str所指向的字符串首地址重复使用后,地址已经不在字符串上,如果继续用将使指针指向字符串的所在存储单元的连续地址上,无法显示正确值,且不能将str清0,如果清0将是把它所指向的地址的清0,这样指针指向的地址就不再是字符串所在的地址了,故需要用一个变量j来记住这个指针的位置。
{
WriteData((str+j));
++j;
Delay_S(2);
}
Delay_S(60); //加上此延时后会出现屏幕闪烁的现象
}
}
str=0;
}
/LCD字符左移滚动显示/
void main()
{
Iinit_Lcd();
while(1)
{
Show_char(1,"My phone 123456789");
}
}
以上就是关于求一个1602液晶显示最简单的汇编程序(在屏上显示一个数字就行了,过程完整)全部的内容,包括:求一个1602液晶显示最简单的汇编程序(在屏上显示一个数字就行了,过程完整)、lcd1602液晶显示的程序、给个单片机液晶1602显示汉字的程序,谢谢等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)