这就是我做测试用的最小系统,单片机是STC89C516,晶振为12M。液晶为蓝底白字的那种1602。
当我们硬件连接错误,或者程序错误时就会出现下图这种情况,就是上排显示16的白色的块(蓝底黑字的液晶则显示的是16个黑块)。
下面我们来驱动1602吧在1602的上排显示“LCD1602 check ok”下排显示“study up”程序中没有用到忙检测,而是用的是延时函数来替代忙检测
#include
#define uint unsigned int//预定义一下
#define uchar unsigned char
sbit rs=P3^5;//1602的数据/指令选择控制线
sbit rw=P3^6;//1602的读写控制线
sbit en=P3^7;//1602的使能控制线
/*P2口接1602的D0~D7,注意不要接错了顺序,我以前可在这上面吃过亏~*/
uchar code table[]="LCD1602 check ok"; //要显示的内容1放入数组table
uchar code table1[]="study up";//要显示的内容2放入数组table1
void delay(uint n)//延时函数
{
uint x,y;
for(x=n;x>0;x--)
for(y=110;y>0;y--);
}
/*********************************************************************************/
void lcd_wcom(uchar com)//1602写命令函数 (单片机给1602写命令)
{//1602接收到命令后,不用存储,直接由HD44780执行并产生相应动作
rs=0;//选择指令寄存器
rw=0;//选择写
P2=com;//把命令字送入P2
delay(5);//延时一小会儿,让1602准备接收数据
en=1;//使能线电平变化,命令送入1602的8位数据口
en=0;
}
void lcd_wdat(uchar dat)//1602写数据函数
{
rs=1;//选择数据寄存器
rw=0;//选择写
P2=dat;//把要显示的数据送入P2
delay(5);//延时一小会儿,让1602准备接收数据
en=1;//使能线电平变化,数据送入1602的8位数据口
en=0;
}
void lcd_init()//1602初始化函数
{
lcd_wcom(0x38);//8位数据,双列,5*7字形
lcd_wcom(0x0c);//开启显示屏,关光标,光标不闪烁
lcd_wcom(0x06);//显示地址递增,即写一个数据后,显示位置右移一位
lcd_wcom(0x01);//清屏
}
void main()//主函数
{
uchar n,m=0;
lcd_init();//液晶初始化
lcd_wcom(0x80);//显示地址设为80H(即00H,)上排第一位(也是执行一条命令)
for(m=0;m<16;m++)//将table[]中的数据依次写入1602显示
{
lcd_wdat(table[m]);
delay(200);
}
lcd_wcom(0x80+0x44); //重新设定显示地址为0xc4,即下排第5位
for(n=0;n<8;n++)//将table1[]中的数据依次写入1602显示
{
lcd_wdat(table1[n]);
delay(200);
}
while(1);//动态停机
}
程序写好后烧写进单片机,现在让我们看看效果吧
这就是显示的效果。
下面让我们来看看如何显示一个自定义的字符吧
我们从CGROM表上可以看到,在表的最左边是一列可以允许用户自定义的CGRAM,从上往下看着是16个,实际只有8个字节可用。它的字符码是00000000-00000111这8个地址,表的下面还有8个字节,但因为这个CGRAM的字符码规定0-2位为地址,3位无效,4-7全为零。因此CGRAM的字符码只有最后三位能用也就是8个字节了。等效为0000X111,X为无效位,最后三位为000-111共8个。
如果我们要想显示这8个用户自定义的字符,操作方法和显示CGROM的一样,先设置DDRAM位置,再向DDRAM写入字符码,例如“A”就是41H。现在我们要显示CGRAM的第一个自定义字符,就向DDRAM写入00000000B(00H),如果要显示第8个就写入00000111(08H),简单吧!
好!现在我们来看怎么向这八个自定义字符写入字模。有个设置CGRAM地址的指令大家还记得吗?赶快再找出来看看。
从这个指令可以看出指令数据的高2位已固定是01,只有后面的6位是地址数据,而这6位中的高3位就表示这八个自定义字符,最后的3位就是字模数据的八个地址了。例如第一个自定义字符的字模地址为01000000-01000111八个地址。我们向这8个字节写入字模数据,让它能显示出“℃”
地址:01000000数据:00010000图示:○○○■○○○○
0100000100000110○○○○○■■○
0100001000001001○○○○■○○■
0100001100001000○○○○■○○○
0100010000001000○○○○■○○○
0100010100001001○○○○■○○■
0100011000000110○○○○○■■○
0100011100000000○○○○○○○○
下面我们写一段程序让这8个自定义字符显示出一个心的图案: (由于上面那个显示程序已经有很详细的注释了,因此这个程序只对与上个程序不同的地方写注释)
#include
#define uint unsigned int
#define uchar unsigned char
sbit rs=P3^5;
sbit rw=P3^6;
sbit en=P3^7;
uchar code table[]={0x03,0x07,0x0f,0x1f,0x1f,0x1f,0x1f,0x1f,
0x18,0x1E,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,
0x07,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,0x1f,
0x10,0x18,0x1c,0x1E,0x1E,0x1E,0x1E,0x1E,
0x0f,0x07,0x03,0x01,0x00,0x00,0x00,0x00,
0x1f,0x1f,0x1f,0x1f,0x1f,0x0f,0x07,0x01,
0x1f,0x1f,0x1f,0x1f,0x1f,0x1c,0x18,0x00,
0x1c,0x18,0x10,0x00,0x00,0x00,0x00,0x00};//心图案
/*ucharcode table1[]={0x10,0x06,0x09,0x08,0x08,0x09,0x06,0x00};//字符℃ */
void delay(uint n)
{
uint x,y;
for(x=n;x>0;x--)
for(y=110;y>0;y--);
}
void lcd_wcom(uchar com)
{
rs=0;
rw=0;
P2=com;
delay(5);
en=1;
en=0;
}
void lcd_wdat(uchar dat)
{
rs=1;
rw=0;
P2=dat;
delay(5);
en=1;
en=0;
}
void lcd_init()
{
lcd_wcom(0x38);
lcd_wcom(0x0c);
lcd_wcom(0x06);
lcd_wcom(0x01);
}
void main()
{
char m=0;
lcd_init();
lcd_wcom(0x40);//设定CGRAM地址
for(m=0;m<64;m++)//将心型代码写入CGRAM中
{
lcd_wdat(table[m]);
}
lcd_wcom(0x85);//设定上排的显示位置
for(m=0;m<4;m++)//显示心型图案的上半部分
{
lcd_wdat(m);
}
lcd_wcom(0xc5);//将显示坐标转移到下排和上排相对应的地方
for(m=4;m<8;m++)//显示心型图案的下半部分
{
lcd_wdat(m);
}
while(1);
}
让我们一起来看看显示的效果吧~~
在绿底黑字液晶模块和蓝底白字液晶模块上分别显示的效果。