
LCD存在静态画面的烧屏问题,就是一个画面时间长了,可能导致某些地方显示留下永久的痕迹!即使换别的画面也可依稀见到以前的画面。
即使没有直接显示看得出的损害,但是长时间显示同一个画面会导致晶体管和液晶分子过热,对液晶单元不好。
但是同时若根本没用到显示器却让显示器显示画面,是有点浪费显示器的寿命的。要知道LCD显示器比较娇气,受液晶单元和灯管的寿命功能影响,其政体寿命较短。
因此建议不要设置屏保而是设置显示器的电源管理,对LCD显示器来讲是最好的
程序只是没有逻辑错误和语法错误,但液晶的控制貌似有些问题。给你一段1602的驱动程序做参考。
#define LCD1602_FLAG
#define LCD1602_PORT P1
#include<reg52h>
#include<stddefh>
#include"dtypeh"
sbit lcd1602_rs=P3^7;
sbit lcd1602_e=P3^5;
sbit lcd1602_rw=P3^6;
sbit lcd1602_busy=P1^7;
/
函数名称:lcd1602_CheckBusy()
函数功能:状态查询
/
void lcd1602_CheckBusy()
{
do
{
lcd1602_busy=1;
lcd1602_rs=0;
lcd1602_rw=1;
lcd1602_e=0;
lcd1602_e=1;
}
while(lcd1602_busy);
}
/
函数名称: lcd1602_WriteCmd()
函数功能:写命令
入口参数:命令字
出口参数:无
/
void lcd1602_WriteCmd(const INT8U cmd)
{
lcd1602_CheckBusy();
lcd1602_rs=0;
lcd1602_rw=0;
lcd1602_e=1;
LCD1602_PORT=cmd;
lcd1602_e=0;
}
/
函数名称:lcd1602_WriteData()
函数功能:写数据
入口参数:c--待写数据
出口参数:无
/
void lcd1602_WriteData(const INT8U c)
{
lcd1602_CheckBusy();
lcd1602_rs=1;
lcd1602_rw=0;
lcd1602_e=1;
LCD1602_PORT=c;
lcd1602_e=0;
}
/
函数名称:lcd1602_Init()
函数功能:初始化LCD
入口参数:无
出口参数:无
/
void lcd1602_Init()
{
lcd1602_WriteCmd(0x38); //显示模式为8位2行57点阵
lcd1602_WriteCmd(0x0f); //display enable,flag enable,flash enable,
lcd1602_WriteCmd(0x06); //flag move to right,screen don't move
lcd1602_WriteCmd(0x01); //clear screen
}
/
函数名称:lcd1602_Display()
函数功能: 字符显示
入口参数:ptr--字符或字符串指针
出口参数:无
说 明:用户可通过以下方式来调用:
1)lcd1602_Display("Hello,world!");
2) INT8U 存储类型 txt[]="要显示的字符串";
或者 INT8U 存储类型 txt[]={'t','x','t',,'\0'};
INT8U ptr;
ptr=&txt;
lcd1602_Display(ptr);
或 lcd1602_Display(txt);
或 lcd1602_Display(&txt);
/
void lcd1602_Display(const INT8U ptr)
{
INT8U data i=0;
INT8U data q;
q=ptr;
lcd1602_WriteCmd(0x80);
while(q!=NULL && (q!='\0') && i<16)
{
lcd1602_WriteData(q);
q++;
i++;
}
lcd1602_WriteCmd(0xc0);
while(q!=NULL && (q!='\0') && i>=16 && i<32)
{
lcd1602_WriteData(q);
q++;
i++;
}
}
是两个子程序。单片机按键切换lcd显示是两个子程序。单片机是一种集成电路芯片,是采用超大规模集成电路技术把具有数据处理能力的中央处理器CPU、随机存储器RAM、只读存储器ROM、多种I/O口和中断系统。
你这个液晶屏不是一般开发板能带有的。大致读了一下:
#include <avr/ioh> #include "boardh" #include "manageh"//调用头文件 #include "scanh" #include "timerh" #include "lcdh"
void sys_init(void);
void bus_reset(void);
int main(void) //主程序
{ uint8_t i = 0; //这里定义了一个计数器,用于指示灯闪烁计时用
sys_init(); //系统初始化,具体程序在下面
int_init(); //又初始化了一个什么东西,先不管
// timer0_init(); //没有用到time0
timer1_init(); //time1初始化,但是就现有程序而言还无法知道计时周期
// timer2_init();
// adc_init();
wdt_disable(); //先禁用看门狗
wdt_enable(WDTO_250MS); //设置看门狗250ms为一周期
wdt_reset(); //复位看门狗
sei(); //禁用中断
// 初始化LCD
lcd_init(); //调用液晶屏初始化程序
while(1) //主程序循环
{ manage(); //应该是处理液晶显示内容用的子程序
if (i & 0x10)
LED_ON(); //0x10个周期后LED闪烁
else
LED_OFF();
i++;
delay_ms(10);
wdt_reset(); //复位看门狗
}
}
// 系统初始化
void sys_init(void) //系统初始化子程序的详细内容(看来只定义了IO口还真不知道这个系统内单片机的定时器如何初始化的)
{
PORTA = 0xf2; //定义IO口
DDRA = 0; // PA7~PA4 键盘输入,PA3: LCD_A0复用电池测量时,输出1
// PA2: 输入,测电池电压, PA1:LCD_CS, PA0:输入,测总线电压?
PORTB = 0xff;
DDRB = 0x00; // PB输出,LCD_DB,PB0~PB4 KEY SCAN
PORTC = 0x80;
DDRC = 0xc0; // PC输出,PC7:LCD_RST,PC6:LCD_LED+,
PORTD = 0x03;
DDRD = 0xf3; // PD7:PWM,PD6:ZL,PD5:ZH,PD4:LED
// PD3返回,PD2开关, PD1:LCD_RD, PD0:LCD_WR
}
void lcd_init(void) //LCD的初始化
{
uint8_t i;
LCD_RST_L(); //应该是复位LCD
delay_ms(2); //延时
LCD_RST_H(); //复位完毕,该复位引脚拉高
for (i=0; i<8; i++) //复位8次看门狗?
{
delay_ms(20);
wdt_reset();
}
write_cmd(0xe2);//system reset 24 //写一个复位命令?
delay_ms(10);
write_cmd(0xeb);//set bias RATIO=1/12 27 =E8 FOR LOW POWER //设置液晶屏偏压
write_cmd(0x81);//set refence voltage ;pm 11 DOUBLE command //设置参考电压
write_cmd(0x83);//set RGB D1 21 //设置RGB方式?
write_cmd(0x2f);//set color mode,65K 22 D4 FOR 256 COLOR 3r-3g-2b//设置彩色显示深度
write_cmd(0x2b);//set line rate,358kps 15 A3//设置啥?不是很明白
write_cmd(0x24);//set n-line 5,TempCompensation 00//设置某种补偿?
write_cmd(0xa2); // 15 set line rate 296k
write_cmd(0xd0);//set partial display 21 D0 BGR//设置一个副屏幕?或许这个彩屏还附带一个小屏幕?
write_cmd(0xd6);//set cen 22 d6 64k color 5-6-5//不明白
write_cmd(0xc0); // 19 LC[2:0]
write_cmd(0xaf); //set display on 18, DC[4:2]=111//仍然不明白
delay_ms(10);
wdt_reset();//重启看门狗
}
void write_cmd(uint8_t cmd)//“写命令”子程序
{
LCD_A0_L();//应该是拉低“命令\数据”引脚的电平
LCD_CS_L(); //片选引脚
PORTB = cmd; //PORTB的8个引脚作为并行口向液晶屏输出数据
DDRB = 0xff; //PORTB的方向定义为输出
LCD_WR_L(); //写LCD引脚电平拉低,看来这程序可能是为了防止被中断或者别的情况打乱,特地发送了两遍
LCD_WR_L();
LCD_WR_H();
LCD_CS_H();
DDRB = 0; //改PORTB为高阻态
//个人觉得后面应该还需要两个语句:LCD_CS_H();和LCD_A0_H();已完成一个“写命令”周期
}
总之,你这块液晶屏应该在市场上不常见。要用这个玩意,你必须有对应这块液晶屏的详细文档、数据手册,否则光这么一个程序很难把这个液晶屏玩转了。
#include<reg51h> //包含单片机寄存器的头文件
#include<intrinsh> //包含_nop_()函数定义的头文件
sbit RS=P2^5; //寄存器选择位,将RS位定义为P25引脚
sbit RW=P2^6; //读写选择位,将RW位定义为P26引脚
sbit EN=P2^7; //使能信号位,将EN位定义为P27引脚
sbit BF=P0^7; //忙碌标志位,,将BF位定义为P07引脚
unsigned char code string[ ]={"ABCD1234"}; //字符串数组,存储待显示的字符串
/
函数功能:延时1ms
(3j+2)i=(3×33+2)×10=1010(微秒),可以认为是1毫秒
/
void delay1ms()
{
unsigned char i,j;
for(i=0;i<10;i++)
for(j=0;j<33;j++)
;
}
/
函数功能:延时若干毫秒
入口参数:n
/
void delay(unsigned char n)
{
unsigned char i;
for(i=0;i<n;i++)
delay1ms();
}
/
函数功能:判断液晶模块的忙碌状态
返回值:result。result=1,忙碌;result=0,不忙
/
unsigned char BusyTest(void)
{
bit result;
RS=0; //根据规定,RS为低电平,RW为高电平时,可以读状态
RW=1;
EN=1; //EN=1,才允许读写
_nop_(); //空 *** 作
_nop_();
_nop_();
_nop_(); //空 *** 作四个机器周期,给硬件反应时间
result=BF; //将忙碌标志电平赋给result
EN=0;
return result;
}
/
函数功能:将模式设置指令或显示地址写入液晶模块
入口参数:dictate
/
void WriteInstruction (unsigned char dictate)
{
while(BusyTest()==1); //如果忙就等待
RS=0; //根据规定,RS和R/W同时为低电平时,可以写入指令
RW=0;
EN=0; //EN置低电平(根据表8-6,写指令时,EN为高脉冲,
//就是让EN从0到1发生正跳变,所以应先置"0"
_nop_();
_nop_(); //空 *** 作两个机器周期,给硬件反应时间
P0=dictate; //将数据送入P0口,即写入指令或地址
_nop_();
_nop_();
_nop_();
_nop_(); //空 *** 作四个机器周期,给硬件反应时间
EN=1; //EN置高电平
_nop_();
_nop_();
_nop_();
_nop_(); //空 *** 作四个机器周期,给硬件反应时间
EN=0; //当EN由高电平跳变成低电平时,液晶模块开始执行命令
}
/
函数功能:指定字符显示的实际地址
入口参数:x
/
void WriteAddress(unsigned char x)
{
WriteInstruction(x|0x80); //显示位置的确定方法规定为"80H+地址码x"
}
/
函数功能:将数据(字符的标准ASCII码)写入液晶模块
入口参数:y(为字符常量)
/
void WriteData(unsigned char y)
{
while(BusyTest()==1);
RS=1; //RS为高电平,RW为低电平时,可以写入数据
RW=0;
EN=0; //EN置低电平(根据表8-6,写指令时,EN为高脉冲,
//就是让EN从0到1发生正跳变,所以应先置"0"
P0=y; //将数据送入P0口,即将数据写入液晶模块
_nop_();
_nop_();
_nop_();
_nop_(); //空 *** 作四个机器周期,给硬件反应时间
EN=1; //EN置高电平
_nop_();
_nop_();
_nop_();
_nop_(); //空 *** 作四个机器周期,给硬件反应时间
EN=0; //当EN由高电平跳变成低电平时,液晶模块开始执行命令
}
/
函数功能:对LCD的显示模式进行初始化设置
/
void LcdInitiate(void)
{
delay(15); //延时15ms,首次写指令时应给LCD一段较长的反应时间
WriteInstruction(0x38); //显示模式设置:16×2显示,5×7点阵,8位数据接口
delay(5); //延时5ms
WriteInstruction(0x38);
delay(5);
WriteInstruction(0x38);
delay(5);
WriteInstruction(0x0f); //显示模式设置:显示开,有光标,光标闪烁
delay(5);
WriteInstruction(0x06); //显示模式设置:光标右移,字符不移
delay(5);
WriteInstruction(0x01); //清屏幕指令,将以前的显示内容清除
delay(5);
}
void main(void) //主函数
{
unsigned char i;
LcdInitiate(); //调用LCD初始化函数
delay(10);
while(1)
{
WriteInstruction(0x01); //清显示:清屏幕指令
WriteAddress(0x00); //设置显示位置为第1行的第1个字。1602型LCD字符显示器在第1个地址显示完毕后,能自动指向下一地址,
//因此只要制定字符串的第1个字符的显示地址即可。
i = 0;
while(string[i] != '\0')
{ //显示字符
WriteData(string[i]);
i++;
delay(150);
}
for(i=0;i<4;i++)
delay(250);
}
}
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);
以上就是关于屏幕保护程序对LCD显示器有益有害全部的内容,包括:屏幕保护程序对LCD显示器有益有害、做一个单片机液晶显示数字的程序,程序没错,但是就是屏幕就是没有显示,麻烦大佬们看一看、单片机按键切换lcd显示等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
微信扫一扫
支付宝扫一扫
评论列表(0条)