如何利用51单片机4*4矩阵键盘和1602液晶实现四位(或者多位)的显示问题?

作者&投稿:危眨 (若有异议请与网页底部的电邮联系)
1602液晶显示器如何通过AT89S51单片机显示4*4矩阵键盘输入的数值,例如用键盘输入100,显示器就显示100~

建议 去 我的领地 5d6d 的 proteus 论坛看看

那里有很多 单片机仿真实例,包括仿真图 和 源码
搜 5d6d 和 proteus 2个关键词

#define LCD1602_FLAG#define LCD1602_PORT P0#include#include#define uchar unsigned char
sbit lcd1602_rs=P2^0;sbit lcd1602_e=P2^2;sbit lcd1602_rw=P2^1;sbit lcd1602_busy=P0^7;
uchar str[]="This is Key Test";uchar keyval,keystr[16];
void delay(uchar a){uchar i;while(a--)for(i=0;i<120;i++);}
/************************************** 函数名称: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 uchar 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 uchar 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行5*7点阵lcd1602_WriteCmd(0x0c); //display enable,flag enable,flash enable,lcd1602_WriteCmd(0x06); //flag move to right,screen don't movelcd1602_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 uchar *ptr,uchar line){uchar data i=0;uchar *data q;
q=ptr;switch(line){case 0:lcd1602_WriteCmd(0x80);while(q!=NULL && (*q!='\0') && i<16){lcd1602_WriteData(*q);q++;i++;}break;case 1:lcd1602_WriteCmd(0xc0);while(q!=NULL && (*q!='\0') && i<16){lcd1602_WriteData(*q);q++;i++;}break;}}
uchar kbscan(void){unsigned char sccode,recode;P3=0x0f; //发0扫描,列线输入if ((P3 & 0x0f) != 0x0f) //有键按下{delay(20); //延时去抖动if ((P3&0x0f)!= 0x0f){sccode = 0xef; //逐行扫描初值while((sccode&0x01)!=0){P3=sccode;if((P3&0x0f)!=0x0f){recode=(P3&0x0f)|0xf0;while((P3&0x0f)!=0x0f);//等待键抬起return((~sccode)+(~recode));}elsesccode=(sccode<<1)|0x01;}}}return 0; //无键按下,返回0}uchar getkey(void){uchar key;key=kbscan();if(key==0){keyval=0xff;return(0);}switch(key){case 0x11:keyval=7;break;case 0x12:keyval=4;break;case 0x14:keyval=1;break;case 0x18:keyval=10;break;case 0x21:keyval=8;break;case 0x22:keyval=5;break;case 0x24:keyval=2;break;case 0x28:keyval=0;break;case 0x41:keyval=9;break;case 0x42:keyval=6;break;case 0x44:keyval=3;break;case 0x48:keyval=11;break;case 0x81:keyval=12;break;case 0x82:keyval=13;break;case 0x84:keyval=14;break;case 0x88:keyval=15;break;default:keyval=0xff;break;}if(keyval!=0xff)return(1);else return(0);}
main(){uchar num;lcd1602_Init();lcd1602_Display(str,0);while(1){if(getkey()){keystr[num]=keyval+0x30;lcd1602_Display(keystr,1);num++;num%=16;}
}}

//按4X4键盘的F键 进入设定状态
//E D 键是前后移动键
//C按键是调整数值按键
#include "reg52.h"
#define uchar unsigned char
sbit RW=P2^1;
sbit RS=P2^0;
sbit E=P2^2;
bit at=0;
uchar code shen[]={"CLOCK!"};
uchar code word[]={0x30,0x31,0x32,0x33,0x34,0x35,0x36,0x37,0x38,0x39,0x3a,0x20};
uchar dispbuf[8],h,m,s,counter;
void delay()
{
uchar i;
for(i=0;i<255;i++);
}

/*******写命令**********/
void lcd_wmc(uchar i)
{

P0=i;
RS=0;
RW=0;
E=0;
delay();
E=1;
}

/*******写数据***********/

void lcd_wmd(uchar i)
{

P0=i;
RS=1;
RW=0;
E=0;
delay();
E=1;
}

/*******初始化液晶*******/
void lcd_init()
{
uchar i;
lcd_wmc(0x01);
lcd_wmc(0x38);
lcd_wmc(0x0c);
lcd_wmc(0x06);
lcd_wmc(0xc9);
for(i=0;i<6;i++)
lcd_wmd(shen[i]);
lcd_wmc(0xc0);
}

/*******更新缓冲区子程序*******/
void newbuf()
{
dispbuf[0]=s%10;
dispbuf[1]=s/10;
dispbuf[3]=m%10;
dispbuf[4]=m/10;
dispbuf[6]=h%10;
dispbuf[7]=h/10;
}

/*******显示子程序**********/
void disp(uchar dispadd)
{
uchar tmp;
lcd_wmc(dispadd);
tmp=dispbuf[7];
tmp=word[tmp];
lcd_wmd(tmp);
tmp=dispbuf[6];
tmp=word[tmp];
lcd_wmd(tmp);
tmp=dispbuf[5];
tmp=word[tmp];
lcd_wmd(tmp);
tmp=dispbuf[4];
tmp=word[tmp];
lcd_wmd(tmp);
tmp=dispbuf[3];
tmp=word[tmp];
lcd_wmd(tmp);
tmp=dispbuf[2];
tmp=word[tmp];
lcd_wmd(tmp);
tmp=dispbuf[1];
tmp=word[tmp];
lcd_wmd(tmp);
tmp=dispbuf[0];
tmp=word[tmp];
lcd_wmd(tmp);
}

/*********************键盘子程序***********************/

uchar keypro(void)
{
uchar scanl,scanh;
P1=0x0f;//先将所有行线拉低
if((P1&0x0f)!=0x0f) //如果列线有变化
{
delay();//延迟一段时间。
if((P1&0x0f)!=0x0f)//如果此时此刻列线还有变化,说明确实有按键按下
{
scanl=0xfe;
while((scanl&0x10)!=0)
{
P1=scanl; //给P1口赋扫描码,每次只拉低一行
if((P1&0xf0)!=0xf0) //如果判断为真,则说明找到了按键按下的行
{
scanh=(P1&0xf0)|0x0f; //计算识别码
return (~scanh)+(~scanl);
}
else scanl=(scanl<<1)|0x01; //否则依次将第二,第三,第四行拉低

}
}
}
return 0;//没有按键 按下 返回0
}

/********************时间调整子程序********************/
void adjustime()
{
uchar k;
static uchar add;
k=keypro();
switch(k)
{
case 0x88: if(!at){add=0xc1;EA=0;lcd_wmc(0xc1);lcd_wmc(0x0f);at=1;}
else {lcd_wmc(0xc0);lcd_wmc(0x0c);at=0;EA=1;} break;

case 0x48: if(at)
{
if(add==0xc1){add=0xc7; lcd_wmc(add);}
else {add=add-3;lcd_wmc(add);}
}
break;

case 0x28: if(at)
{
if(add==0xc7) {add=0xc1;lcd_wmc(add);}
else {add=add+3;lcd_wmc(add);}
}
break;

case 0x18: if(at)
{
if(add==0xc1) h++;
if(h==24)
h=0;
if(add==0xc4) m++;
if(m==60)
m=0;
if(add==0xc7) s++;
if(s==60)
s=0;
newbuf();
disp(0xc0);
lcd_wmc(add);
}
break;
default: break;
}
if(k!=0)
{
while((P1&0xf0)!=0xf0)
P1=0xf0;
}

}

/*********************初始化子程序**********************/
void init()
{

TMOD=0x01;
TH0=0x4c;
TL0=0x00;
EA=1;
ET0=1;
TR0=1;
counter=0;
h=12;m=0;s=0;
dispbuf[2]=10;
dispbuf[5]=10;
}

/***************************主程序************************/

void main(void)
{
init();
lcd_init();

while(1)
{
adjustime();
if(!at)
{
//闪烁
if(counter<10)
{
dispbuf[2]=10;
dispbuf[5]=10;
}
else
{
dispbuf[2]=11;
dispbuf[5]=11;
}
//更新显示缓冲区及调用显示程序
if(counter==0)
{
newbuf();
disp(0xc0);
}
else if(counter==10)
disp(0xc0);
}
}
}

/*************************定时器0的中断**********************/
void Time0() interrupt 1 using 2 //再次强调中断子程序执行时间越短越好
{
TH0=(65536-46075)/256;
TL0=(65536-46075)%256;
counter++;
if(counter==20)
{
s++;
counter=0;
if(s==60)
{
m++;
s=0;
if(m==60)
{
h++;
m=0;
if(h==24)
h=0;
}
}

}
}
这是一个数字时钟的程序,用矩阵键盘调时

4*4矩阵键盘用行列扫描方式获得键值,1602有固定的驱动时序,按时序操作并发数据到1602就可以了

用什么语言编程?
什么具体问题啊?

51单片机 c语言编程 4*4矩阵键盘如何实现等待按键松开功能?
答:0xf7:h=3;break;}switch(l){case 0xef:l=0;break;//1110 1111case 0xdf:l=1;break;case 0xbf:l=2;break;case 0x7f:l=3;break;}P1=0xf0;/***关键问题解决如下***/while(P1!=0xf0){while(P1!=0xf0);//等待放开按键delayms(50);//防止抖动}return(h*4+l);} ...

51单片机。4*4矩阵键盘扫描
答:矩阵键盘扫描是分两次的。比方说,51单片机,默认各口是高电平,首先,比方图中P1.0-P1.3给低电平(注:你的图太小,我就把接矩阵键盘的各口当作P1.0-P1.7),若图中无键被按下,P1.4-P1.7扫描出来的就都是高电平,若有键被按下,则P1.4-P1.7某根线就会是低电平,但是比方说P...

急求一个基于89C51单片机,用4*4矩阵键盘输入数据,并在LCD1602上显示出来...
答:define LCD1602_FLAG#define LCD1602_PORT P0#include<reg51.h>#include<stddef.h>#define uchar unsigned char sbit lcd1602_rs=P2^0;sbit lcd1602_e=P2^2;sbit lcd1602_rw=P2^1;sbit lcd1602_busy=P0^7;uchar str[]="This is Key Test";uchar keyval,keystr[16];void delay(uchar a){...

51单片机4*4矩阵键盘用C语言如何设置成密码开锁键?
答:include<stdio.h> include<reg51.h> define uchar unsigned char uchar key;int i=0;char b[4]={'0','0','0','0'};//输入的密码放这里和初始的比较来判断是否密码正确 void keyscan()//扫描按键是否按下 { uchar temp;P3=0xfe;//扫描第一行 temp=P3;temp=temp&0xf0;if(temp!=0x...

单片机C51 4×4矩阵式键盘识别技术
答:include <reg52.h> include <intrins.h> sbit IRIN = P3 ^ 2; //红外接收器数据线 sbit RELAY = P3 ^ 6; //继电器驱动线 unsigned char code scan_bit[8]={ 0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f };unsigned int aa,bb,cc,key,shu,temp1;unsigned char temp,n;unsigned ...

51单片机 如何4*4矩阵键盘输入按键 继电器输出信号
答://开中断一 P3.3 IT1=1; //下降沿触发 ET1=1;EA=1;...} 注:图中的键盘接法都是独立按键接发,矩阵键盘没有见过能直接触发中断的。

C51单片机用IO口做为几个按键的控制,现在想用4X4键盘控制,请问下面显示...
答:// 4*4矩阵键盘 / 连接:--- P1.0 P1.1 P1.2 P1.3 p1.4 S1 S5 S9 S13 P1.5 S2 S6 S10 S14 P1.6 S3 S7 S11 S15 P1.7 S4 S8 S12 S16 --- 实现原理:当高电平端口与低电平端口短接时,高电平端口会被拉成低电平 --- 算法分析:1,检查是否有键按下:P1=f0h (即0-...

51单片机,4*4矩阵键盘,八位数码管
答:51单片机,4*4矩阵键盘,八位数码管,仿真实例可以参考一下。include<reg51.h> define uchar unsigned char uchar temp;int key1,key,disbuf;// 此表为 LED 的字模 0 1 2 3 4 5 6 7 8 9 a b c d e f unsigned char code LED7Code[] = {0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x...

求简单的51系列单片机控制4*4矩阵键盘显示电路图 带上程序
答:/*功能:矩阵按键识别,每按一次响一声,用数码管显示0——F键值 */ include <reg52.h> include <intrins.h> define uchar unsigned char define uint unsigned int sbit sound=P3^6;uchar code tab[]= { 0xC0,/*0*/ 0xF9,/*1*/ 0xA4,/*2*/ 0xB0,/*3*/ 0x99,/*4*/ 0x92,/*...

51单片机按键矩阵4X4求指点 不明白他是啥意思
答:P3端口4根线(高4位)作为输入,另4根线作为输出,作为输入的4根线输出为1(51单片机引脚作为输入时先写1),如果去读取结果当然为1,但有按键接在上面和4根输出线相连,并且这4根输出线轮流输出低电平,有按键按下时,这个低电平就传到了输入引脚,这时你读这4个引脚的状态,就不全是1了,根据哪个...