标题:
51单片机开发板Mini12864汉字与字符的显示方法 附源程序
[打印本页]
作者:
张小超
时间:
2019-7-26 20:32
标题:
51单片机开发板Mini12864汉字与字符的显示方法 附源程序
/************************************************************************
对于某中科技开发板上12864的使用,其自带程序中只有关于汉字的程序,没有关于字符的,所以我在使用其显示汉字与字符时屏幕会出现乱码,经过我仔细的研究发现,其汉字和字符是不能直接用一个函数显示的,因为汉字是以16*16的形式存储,而字符是以8*16的方式,所以如果将字符取模直接存入字库使用就会出现乱码,但是如果重新再为字符创建一个对应的函数又过于麻烦,所以,我用的方法是将字符以16*16的形式存于字库中,但是如果所以只存一个字符格式,屏幕也会出现乱码,所以采用重叠的形式进行储存,比如存入"*",就需要用"**"的方式,对于数字可以在前面加一个0,比喻"1"可以存为"01",这个可以根据自己所需要的数据而定,若果存储的数据是偶数为,就可以以"十位+个位"的形式,这里不多讲,看程序一眼就明白了。
同时,在存储过程中,会有一个问题,因为程序在扫描的时候是分成四部扫描的,但是取模出来的字符与其扫描的顺序有所不同,所以在我们取模后需要进行一定的调整,比如对数字1取模为0x01,0x03,对数字2取模为0x02,0x04,在字库表达12是就不能直接写成0x01,0x03,0x02,0x04,这就会出现数字上下结构成为了左右结构,所以我们在写12的字库时要写成0x01,0x02,0x03,0x04(注:字符取模出来的16进制数有16位,所以在交换数据位置时是以8位为一个单位进行交换的),以下是基于一种红外密码锁的显示的主程序
************************************************************************/
单片机源程序如下:
#include<reg51.h>
#include"st7565.h"
/********************************************************************
连线说明:
lcd12864:
CS = P1^1;//片选,低电平有效
RESET = P1^2;//复位,低电平有效
RS = P3^5;//数据/指令选择,高电平为数据,低电平为指令
WR = P3^7;//写使能,低电平有效
RD = P3^6;//读使能,低电平有效
P0->D0-D7
红外管:
IRIN=P3^2;
蜂鸣器:
ALAM = P1^7;
AT24C02:
Scl=P2^1; //24C02串行时钟
Sda=P2^0; //24C02串行数据
LED(开锁装置)
LED = P1^6; //开锁
**********************************************************************/
sbit Scl=P2^1; //24C02串行时钟
sbit Sda=P2^0; //24C02串行数据
sbit IRIN=P3^2; //红外
sbit ALAM = P1^7; //报警
sbit LED = P1^6; //开锁
bit pass=0; //密码正确标志
bit ReInputEn=0; //重置输入允许标志
bit s3_keydown=0; //3秒按键标志位
bit key_disable=0; //锁定键盘标志
unsigned char IrValue[6];
unsigned char Time,Timesec,Timemin,Timehour;
//unsigned char sec=0,min=0,hour=23;
unsigned char countt0,second; //t0中断计数器,秒计数器
unsigned char Time;
void Delay5Ms(void); //声明延时函数
//液晶显示数据数组
unsigned char code start_line[] = {"榕轩:0B8006"};
unsigned char code name[] = {"未解锁"}; //显示名称
unsigned char code Correct[] = {"输入正确"}; //输入正确
unsigned char code Error[] = {"输入错误"}; //输入错误
unsigned char code codepass[] = {"密码通过"};
unsigned char code LockOpen[] = {"未上锁"}; //OPEN
unsigned char code SetNew[] = {"设置新密码"};
unsigned char code Input[] = {"请输入密码:"}; //INPUT
unsigned char code ResetOK[] = {"密码重置完成"};
unsigned char code initword[] = {"初始化中"};
unsigned char code Er_try[] = {"密码错误,请重输"};
unsigned char code again[] = "请在次输入密码";
unsigned char code zhuang[] = "状态:";
unsigned char InputData[6]; //输入密码暂存区
unsigned char CurrentPassword[6]={0,0,0,0,0,0}; //读取EEPROM密码暂存数组
unsigned char TempPassword[6];
unsigned char N=0; //密码输入位数记数
unsigned char ErrorCont; //错误次数计数
unsigned char CorrectCont; //正确输入计数
unsigned char ReInputCont; //重新输入计数
unsigned char code initpassword[6]={0,0,0,0,0,0}; //输入管理员密码后将密码初始为000000
unsigned char code adminpassword[6]={1,3,1,4,2,0}; //输入管理员密码后将密码初始为000000
//---声明一个全局变量---//
void Delay10ms(unsigned int c);
void Clear();
/*******************************************************************************
* 函数名 : DelayMs()
* 函数功能 : 延时
* 输入 : x
* 输出 : 无
*******************************************************************************/
void DelayMs(unsigned int x) //0.14ms误差 0us
{
unsigned char i;
while(x--)
{
for (i = 0; i<13; i++)
{}
}
}
//=====================5ms延时==============================
void Delay5Ms(void)
{
unsigned int TempCyc = 5552;
while(TempCyc--);
}
//===================400ms延时==============================
void Delay400Ms(void)
{
unsigned char TempCycA = 5;
unsigned int TempCycB;
while(TempCycA--)
{
TempCycB=7269;
while(TempCycB--);
}
}
//=============================================================================================
//================================24C02========================================================
//=============================================================================================
void mDelay(uint t) //延时
{
uchar i;
while(t--)
{
for(i=0;i<125;i++)
{;}
}
}
void Nop(void) //空操作
{
_nop_(); //仅作延时用一条语句大约1us
_nop_();
_nop_();
_nop_();
}
//==============================开锁功能==============================
void WORK()
{
LED=~LED;
Delay400Ms();
Delay400Ms();
Delay400Ms();
Lcd12864_Write16CnCHAR(2, 40, name);
}
/*****24c02程序参照24c02时序图*****/
/*起始条件*/
void Start(void)
{
Sda=1;
Scl=1;
Nop();
Sda=0;
Nop();
}
/*停止条件*/
void Stop(void)
{
Sda=0;
Scl=1;
Nop();
Sda=1;
Nop();
}
/*应答位*/
void Ack(void)
{
Sda=0;
Nop();
Scl=1;
Nop();
Scl=0;
}
/*反向应答位*/
void NoAck(void)
{
Sda=1;
Nop();
Scl=1;
Nop();
Scl=0;
}
/*发送数据子程序,Data为要求发送的数据*/
void Send(uchar Data)
{
uchar BitCounter=8;
uchar temp;
do
{
temp=Data; //将待发送数据暂存temp
Scl=0;
Nop();
if((temp&0x80)==0x80) //将读到的数据&0x80
Sda=1;
else
Sda=0;
Scl=1;
temp=Data<<1; //数据左移
Data=temp; //数据左移后重新赋值Data
BitCounter--; //该变量减到0时,数据也就传送完成了
}
while(BitCounter); //判断是否传送完成
Scl=0;
}
/*读一字节的数据,并返回该字节值*/
uchar Read(void)
{
uchar temp=0;
uchar temp1=0;
uchar BitCounter=8;
Sda=1;
do
{
Scl=0;
Nop();
Scl=1;
Nop();
if(Sda) //数据位是否为1
temp=temp|0x01; //为1 temp的最低位为1(|0x01,就是将最低位变为1)
else //如果为0
temp=temp&0xfe; //temp最低位为0(&0xfe(11111110)最低位就是0)
if(BitCounter-1) //BitCounter减1后是否为真
{
temp1=temp<<1; //temp左移
temp=temp1;
}
BitCounter--; //BitCounter减到0时,数据就接收完了
}
while(BitCounter); //判断是否接收完成
return(temp);
}
void WrToROM(uchar Data[],uchar Address,uchar Num)
{
uchar i;
uchar *PData;
PData=Data;
for(i=0;i<Num;i++)
{
Start();
Send(0xa0);
Ack();
Send(Address+i);
Ack();
Send(*(PData+i));
Ack();
Stop();
mDelay(20);
}
}
void RdFromROM(uchar Data[],uchar Address,uchar Num)
{
uchar i;
uchar *PData;
PData=Data;
for(i=0;i<Num;i++)
{
Start();
Send(0xa0);
Ack();
Send(Address+i);
Ack();
Start();
Send(0xa1);
Ack();
*(PData+i)=Read();
Scl=0;
NoAck();
Stop();
}
}
unsigned char cod_red(unsigned char m)
{
unsigned char k;
switch(m)
{
case (0x16): k=0;break;
case (0x0c): k=1;break;
case (0x18): k=2;break;
case (0x5e): k=3;break;
case (0x08): k=4;break;
case (0x1c): k=5;break;
case (0x5a): k=6;break;
case (0x42): k=7;break;
case (0x52): k=8;break;
case (0x4a): k=9;break;//数字键
case (0x40): k='*';break;//左移一位
case (0x43): k='#';break;//右移一位
case (0x46): k='M';break;//时间设置
case (0x47): k='B';break;//关掉声音
case (0x44): k='K';break;//开锁
case (0x07): k='E';break;//确认
case (0x19): k='D';break;//重置密码
case (0x45): k='A';break; //未开发键
case (0x15): k='V';break;
case (0x09): k='O';break;
case (0x0D): k='U';break;
}
IrValue[2]=0x00;
return(k);
}
//=====================按键检测并返回按键值===============================
unsigned char red_num(void)
{
if((IrValue[2]==0x16)||(IrValue[2]==0x0c)||(IrValue[2]==0x18)||(IrValue[2]==0x5e)||(IrValue[2]==0x08)||(IrValue[2]==0x1c)||(IrValue[2]==0x5a)||(IrValue[2]==0x42)||(IrValue[2]==0x52)||(IrValue[2]==0x4a)||(IrValue[2]==0x40)||(IrValue[2]==0x43)||(IrValue[2]==0x19))
{
return IrValue[2];
}
else return 0; //无键按下时返回0
}
//=======================一声提示音,表示有效输入========================
void OneAlam(void)
{
ALAM=0;
Delay5Ms();
ALAM=1;
}
//========================二声提示音,表示操作成功========================
void TwoAlam(void)
{
ALAM=0;
Delay5Ms();
ALAM=1;
Delay5Ms();
ALAM=0;
Delay5Ms();
ALAM=1;
}
//========================三声提示音,表示错误========================
void ThreeAlam(void)
{
ALAM=0;
Delay5Ms();
ALAM=1;
Delay5Ms();
ALAM=0;
Delay5Ms();
ALAM=1;
Delay5Ms();
ALAM=0;
Delay5Ms();
ALAM=1;
}
//=======================显示提示输入=========================
void DisplayChar(void)
{
if(pass==1)
{
//DisplayListChar(0,1,LockOpen);
Clear();
Lcd12864_Write16CnCHAR(2, 40, LockOpen);
WORK();
}
else
{
if(N==0)
{
//DisplayListChar(0,1,Error);
Clear();
Lcd12864_Write16CnCHAR(4, 40, Error);
}
else
{
//DisplayListChar(0,1,start_line);
Clear();
}
}
}
//========================重置密码==================================================
//==================================================================================
void ResetPassword(void)
{
unsigned char i;
if(pass==0)
{
pass=0;
DisplayChar(); //显示错误
ThreeAlam(); //没开锁时按下重置密码报警3声
}
else //开锁状态下才能进行密码重置程序
{
if(ReInputEn==1) //开锁状态下,ReInputEn置1,重置密码允许
{
if(N==6) //输入6位密码
{
ReInputCont++; //密码次数计数
if(ReInputCont==2) //输入两次密码
{
for(i=0;i<6;)
{
if(TempPassword[i]==InputData[i]) //将两次输入的新密码作对比
i++;
else //如果两次的密码不同
{
//DisplayListChar(0,1,Error);
Clear();
Lcd12864_Write16CnCHAR(4, 40, Error);
ThreeAlam(); //错误提示
pass=0; //关锁
ReInputEn=0; //关闭重置功能,
ReInputCont=0;
DisplayChar();
break;
}
}
if(i==6)
{
//DisplayListChar(0,1,ResetOK);
Clear();
Lcd12864_Write16CnCHAR(4, 16, ResetOK);
TwoAlam(); //操作成功提示
WrToROM(TempPassword,0,6); //将新密码写入24C02存储
ReInputEn=0;
}
ReInputCont=0;
CorrectCont=0;
Delay400Ms();
Clear();
Lcd12864_Write16CnCHAR(2, 40, name);
}
else //输入一次密码时
{
OneAlam();
//DisplayListChar(0, 1, again); //显示再次输入一次
Clear();
Lcd12864_Write16CnCHAR(4, 8, again);
for(i=0;i<6;i++)
{
TempPassword[i]=InputData[i]; //将第一次输入的数据暂存起来
}
}
N=0; //输入数据位数计数器清零
}
}
}
}
//=======================输入密码错误超过三过,报警并锁死键盘======================
void Alam_KeyUnable(void)
{
IRIN=0;
{
ALAM=~ALAM; //蜂鸣器一直闪烁鸣响
Delay5Ms();
}
IRIN=1;
}
//=======================取消所有操作============================================
void Cancel(void)
{
unsigned char i;
//DisplayListChar(0, 1, start_line);
Clear();
Lcd12864_Write16CnCHAR(2, 40, name);
TwoAlam(); //提示音
for(i=0;i<6;i++)
{
InputData[i]=0; //将输入密码清零
}
LED=0; //关闭锁
ALAM=1; //报警关
pass=0; //密码正确标志清零
ReInputEn=0; //重置输入充许标志清零
ErrorCont=0; //密码错误输入次数清零
CorrectCont=0; //密码正确输入次数清零
ReInputCont=0; //重置密码输入次数清零
s3_keydown=0;
key_disable=0; //锁定键盘标志清零
N=0; //输入位数计数器清零
}
//==========================确认键,并通过相应标志位执行相应功能===============================
void Ensure(void)
{
unsigned char i,j,m=150;
RdFromROM(CurrentPassword,0,6); //从24C02里读出存储密码
if(N==6)
{
if(ReInputEn==0) //重置密码功能未开启
{
for(i=0;i<6;)
{
if(CurrentPassword[i]==InputData[i]) //判断输入密码和24c02中的密码是否相同
{
i++; //相同一位 i就+1
}
else //如果有密码不同
{
ErrorCont++; //错误次数++
if(ErrorCont==3) //错误输入计数达三次时,报警并锁定键盘
{
Clear();
Lcd12864_Write16CnCHAR(4, 40, Error);
Clear();
Lcd12864_Write16CnCHAR(4, 0, Input);
do
Alam_KeyUnable(),Delay400Ms();
while(m--);
}
else //错误次数小于3次时,锁死键盘3秒,然后重新可以输入
{
TR1=1; //开启定时
key_disable=1; //锁定键盘
pass=0; //pass位清零
break; //跳出
}
}
}
if(i==6) //密码输入对时
{
if((InputData[0]==adminpassword[0])&&(InputData[1]==adminpassword[1])&&(InputData[2]==adminpassword[2])&&(InputData[3]==adminpassword[3])&&(InputData[4]==adminpassword[4])&&(InputData[5]==adminpassword[5]))
{
WrToROM(initpassword,0,6); //强制将初始密码写入24C02存储
Clear();
Lcd12864_Write16CnCHAR(2, 40, initword);//初始化密码
TwoAlam(); //成功提示音
Delay400Ms(); //延时400ms
TwoAlam(); //成功提示音
Lcd12864_Init();
Lcd12864_ClearScreen();
Lcd12864_Write16CnCHAR(0, 0,start_line);
Delay10ms(10);
Lcd12864_Write16CnCHAR(2, 40,name);
Delay10ms(10);
Lcd12864_Write16CnCHAR(4, 0, Input);
Delay10ms(10);
N=0; //输入位数计数器清零
}
else
{
CorrectCont++; //输入正确变量++
if(CorrectCont==1) //正确输入计数,当只有一次正确输入时,开锁
{
//DisplayListChar(0,1,LockOpen);
Clear();
Lcd12864_Write16CnCHAR(2, 40, LockOpen);
TwoAlam(); //操作成功提示音
WORK(); //开锁
pass=1; //置正确标志位
TR1=1; //开启定时
for(j=0;j<6;j++) //将输入清除
{
InputData[i]=0; //开锁后将输入位清零
}
}
else //当两次正确输入时,开启重置密码功能
{
Clear();
Lcd12864_Write16CnCHAR(4, 40, SetNew);
TwoAlam(); //操作成功提示
ReInputEn=1; //允许重置密码输入
CorrectCont=0; //正确计数器清零
}
}
}
else //=========================当第一次使用或忘记密码时可以用131420对其密码初始化============
{
if((InputData[0]==adminpassword[0])&&(InputData[1]==adminpassword[1])&&(InputData[2]==adminpassword[2])&&(InputData[3]==adminpassword[3])&&(InputData[4]==adminpassword[4])&&(InputData[5]==adminpassword[5]))
{
WrToROM(initpassword,0,6); //强制将初始密码写入24C02存储
Clear();
Lcd12864_Write16CnCHAR(2, 40, initword);
TwoAlam(); //成功提示音
Delay400Ms(); //延时400ms
TwoAlam(); //成功提示音
Lcd12864_Init();
Lcd12864_ClearScreen();
Lcd12864_Write16CnCHAR(0, 0,start_line);
Delay10ms(10);
Lcd12864_Write16CnCHAR(2, 40,name);
Delay10ms(10);
Lcd12864_Write16CnCHAR(4, 0, Input);
Delay10ms(10);
N=0; //输入位数计数器清零
}
else //密码输入错误
{
Clear();
Lcd12864_Write16CnCHAR(4, 40, Error);
ThreeAlam(); //错误提示音
pass=0;
}
}
}
else //当已经开启重置密码功能时,而按下开锁键,
{
//DisplayListChar(0,1,Er_try);
Clear();
Lcd12864_Write16CnCHAR(4, 0, Er_try);
ThreeAlam(); //错误提示音
}
}
else //密码没有输入到6位时,按下确认键时
{
//DisplayListChar(0,1,Error);
Clear();
Lcd12864_Write16CnCHAR(4, 40, Error);
ThreeAlam(); //错误提示音
pass=0;
}
N=0; //将输入数据计数器清零,为下一次输入作准备
}
/*******************************************************************************
* 函 数 名 : main
* 函数功能 : 主函数
* 输 入 : 无
* 输 出 : 无
*******************************************************************************/
void main()
{
unsigned char NUM,RED;
unsigned char j;
P1=0xFF; //P1口复位
IT0=1;//下降沿触发
EX0=1;//打开中断0允许
EA=1; //打开总中断
IRIN=1;//初始化端口
TMOD=0x11; //定义工作方式
TL1=0xB0;
TH1=0x3C; //定时器赋初值
EA=1; //打开中断总开关
ET1=1; //打开中断允许开关
TR1=0; //打开定时器开关
Delay400Ms(); //启动等待,等LCM讲入工作状态
Lcd12864_Init();
Lcd12864_ClearScreen();
Lcd12864_Write16CnCHAR(0, 0,start_line);
Delay10ms(10);
Lcd12864_Write16CnCHAR(2,0,zhuang);
Lcd12864_Write16CnCHAR(2, 40,name);
Delay10ms(10);
Lcd12864_Write16CnCHAR(4, 0, Input);
Delay10ms(10);
N=0; //初始化数据输入位数
while(1) //进入循环
{
if(key_disable==1) //锁定键盘标志为1时
Alam_KeyUnable(); //报警键盘锁
else
ALAM=1; //关报警
RED=red_num();
if(RED!=0) //当有按键按下时
{
if(key_disable==1) //锁定键盘标志为1时
{
second=0; //秒清零
}
else //没有锁定键盘时
{
NUM=cod_red(IrValue[2]); //根据按键的位置将其编码,编码值赋值给NUM
{
switch(NUM) //判断按键值
{
case ('A'): ; break;
case ('B'): ; break;
case ('M'):
Clear();
Lcd12864_Write16CnCHAR(2, 40, name);
if(N>=1) N--; OneAlam(); //按键提示音
//DisplayOneChar(6+N,1,'*');
for(j=0;j<N*16;j)
{
Lcd12864_Write16CnCHAR(6, j, "**"); //但不显示实际数字,用*代替
j=j+16;
}
InputData[N]=N+4;
break; //ABC是无定义按键
case ('D'): ResetPassword(); break; //重新设置密码
case ('*'): Cancel(); break; //取消当前输入
case ('#'): Ensure(); break; //确认键,
default: //如果不是功能键按下时,就是数字键按下
{
//DisplayListChar(0,1,Input);
if(N<6) //当输入的密码少于6位时,接受输入并保存,大于6位时则无效。
{
Clear();
Lcd12864_Write16CnCHAR(4, 0, Input);
OneAlam(); //按键提示音
for(j=0;j<=N*16;j)
{
Lcd12864_Write16CnCHAR(6, j, "**"); //显示位数随输入增加而增加
j=j+16; //但不显示实际数字,用*代替
}
InputData[N]=NUM; //将数字键的码赋值给InputData[]数组暂存
N++; //密码位数加
}
else //输入数据位数大于6后,忽略输入
{
N=6; //密码输入大于6位时,不接受输入
break;
}
}
}
}
}
} RED=0;
}
}
/*******************************************************************************
* 函 数 名 : Delay10ms
* 函数功能 : 延时函数,延时10ms
* 输 入 : 无
* 输 出 : 无
*******************************************************************************/
void Delay10ms(unsigned int c) //误差 0us
{
unsigned char a,b;
for(;c>0;c--)
for(b=38;b>0;b--)
for(a=130;a>0;a--);
}
void Clear()
{
Lcd12864_ClearScreen();
Lcd12864_Write16CnCHAR(0, 0,start_line);
Lcd12864_Write16CnCHAR(2,0,zhuang);
}
void time1_int(void) interrupt 3 //定时器T0
{
TL1=0xB0;
TH1=0x3C; //定时器重新赋初值
//TR0=1;
countt0++; //计时变量加,加1次时50ms
if(countt0==20) //加到20次就是1s
{
countt0=0; //变量清零
second++; //秒加
if(pass==1) //开锁状态时
{
if(second==1) //秒加到1s时
{
TR1=0; //关定时器
TL1=0xB0;
TH1=0x3C; //再次赋初值
second=0; //秒清零
}
}
else //不在开锁状态时
{
if(second==3) //秒加到3时
{
TR1=0; //关闭定时器
second=0; //秒清零
key_disable=0; //锁定键盘清零
s3_keydown=0;
TL1=0xB0;
TH1=0x3C; //重新赋初值
}
else
TR1=1; //打开定时器
}
}
}
void ReadIr() interrupt 0
{
unsigned char j,k;
unsigned int err;
Time=0;
DelayMs(70);
if(IRIN==0) //确认是否真的接收到正确的信号
{
err=1000; //1000*10us=10ms,超过说明接收到错误的信号
/*当两个条件都为真是循环,如果有一个条件为假的时候跳出循环,免得程序出错的时
侯,程序死在这里*/
while((IRIN==0)&&(err>0)) //等待前面9ms的低电平过去
{
DelayMs(1);
err--;
}
if(IRIN==1) //如果正确等到9ms低电平
{
err=500;
while((IRIN==1)&&(err>0)) //等待4.5ms的起始高电平过去
{
DelayMs(1);
err--;
}
for(k=0;k<4;k++) //共有4组数据
{
for(j=0;j<8;j++) //接收一组数据
{
err=60;
while((IRIN==0)&&(err>0))//等待信号前面的560us低电平过去
// while (!IRIN)
{
DelayMs(1);
err--;
}
err=500;
while((IRIN==1)&&(err>0)) //计算高电平的时间长度。
{
DelayMs(1);//0.14ms
Time++;
err--;
if(Time>30)
{
EX0=1;
return;
}
}
IrValue[k]>>=1; //k表示第几组数据
if(Time>=8) //如果高电平出现大于565us,那么是1
{
IrValue[k]|=0x80;
}
Time=0; //用完时间要重新赋值
}
}
}
if(IrValue[2]!=~IrValue[3])
{
return;
}
}
}
复制代码
全部资料51hei下载地址:
LCD12864液晶.7z
(3.59 MB, 下载次数: 16)
2019-7-27 02:32 上传
点击文件名下载附件
下载积分: 黑币 -5
欢迎光临 (http://www.51hei.com/bbs/)
Powered by Discuz! X3.1