视频链接:https://www.bilibili.com/video/BV1at4y117Eq
#include< reg51.h >
#define uchar unsigned char
#define uint unsigned int
#define LCD1602_DATAPINS P0
sbit LCD1602_E=P2^2;
sbit LCD1602_RW=P2^1;
sbit LCD1602_RS=P2^0;
sbit KEY = P3^0;
sbit h1 = P1^0;
sbit h2 = P1^1;
sbit h3 = P1^2;
sbit h4 = P1^3;
sbit l1 = P1^4;
sbit l2 = P1^5;
sbit l3 = P1^6;
sbit l4 = P1^7;
/*******************************************************************************
* 函 数 名 : Lcd1602_Delay1ms
* 函数功能 : 延时函数,延时1ms
* 输 入 : c
* 输 出 : 无
* 说 名 : 该函数是在12MHZ晶振下,12分频单片机的延时。
*******************************************************************************/
void Lcd1602_Delay1ms(uint c) //误差 0us
{
uchar a,b;
for (; c>0; c--)
{
for (b=199;b>0;b--)
{
for(a=1;a>0;a--);
}
}
}
/*******************************************************************************
* 函 数 名 : LcdWriteCom
* 函数功能 : 向LCD写入一个字节的命令
* 输 入 : com
* 输 出 : 无
*******************************************************************************/
void LcdWriteCom(uchar com) //写入命令
{
LCD1602_E = 0; //使能
LCD1602_RS = 0; //选择发送命令
LCD1602_RW = 0; //选择写入
LCD1602_DATAPINS = com; //放入命令
Lcd1602_Delay1ms(1); //等待数据稳定
LCD1602_E = 1; //写入时序
Lcd1602_Delay1ms(5); //保持时间
LCD1602_E = 0;
}
/*******************************************************************************
* 函 数 名 : LcdWriteData
* 函数功能 : 向LCD写入一个字节的数据
* 输 入 : dat
* 输 出 : 无
*******************************************************************************/
void LcdWriteData(uchar dat) //写入数据
{
LCD1602_E = 0; //使能清零
LCD1602_RS = 1; //选择输入数据
LCD1602_RW = 0; //选择写入
LCD1602_DATAPINS = dat; //写入数据
Lcd1602_Delay1ms(1);
LCD1602_E = 1; //写入时序
Lcd1602_Delay1ms(5); //保持时间
LCD1602_E = 0;
}
/*******************************************************************************
* 函 数 名 : LcdInit()
* 函数功能 : 初始化LCD屏
* 输 入 : 无
* 输 出 : 无
*******************************************************************************/
void LcdInit() //LCD初始化子程序
{
LcdWriteCom(0x38); //开显示
LcdWriteCom(0x0c); //开显示不显示光标
LcdWriteCom(0x06); //写一个指针加1
LcdWriteCom(0x01); //清屏
LcdWriteCom(0x80); //设置数据指针起点
}
//按指定位置显示一个字符
void DisplayOneChar(unsigned char X, unsigned char Y, unsigned char DData)
{
Y &= 0x1;
X &= 0xF; //限制X不能大于15,Y不能大于1
if (Y) X |= 0x40; //当要显示第二行时地址码+0x40;
X |= 0x80; // 算出指令码
LcdWriteCom(X); //这里不检测忙信号,发送地址码
LcdWriteData(DData);
}
//按指定位置显示一串字符
void DisplayListChar(unsigned char X, unsigned char Y, unsigned char code *DData)
{
unsigned char ListLength;
ListLength = 0;
Y &= 0x1;
X &= 0xF; //限制X不能大于15,Y不能大于1
while (DData[ListLength]>=0x20) //若到达字串尾则退出
{
if (X <= 0xF) //X坐标应小于0xF
{
DisplayOneChar(X, Y, DData[ListLength]); //显示单个字符
ListLength++;
X++;
}
}
}
void KEY_scan()
{
if(KEY==0)
{
Lcd1602_Delay1ms(20);
if(KEY==0)
{
while(KEY==0);
}
}
}
void key_scan1()
{
h1=1;
h2=1;
h3=1;
h4=1;
l1=1;
l2=1;
l3=1;
l4=1;
h1=0;
if(l1==0 && h1==0 && h2==1 && l2==1 && l3==1 && l4==1 ) //1
{
while(l1==0);
DisplayOneChar(0,0,'0');
}else if(l1==1 && h1==0 && h2==1 && l2==0 && l3==1 && l4==1 ) //2
{
while(l2==0);
DisplayOneChar(0,0,'1');
}else if(l1==1 && h1==0 && h2==1 && l2==1 && l3==0 && l4==1)//3
{
while(l3==0);
DisplayOneChar(0,0,'2');
}else if(l1==1 && h1==0 && h2==1 && l2==1 && l3==1 && l4==0)//4
{
while(l4==0);
DisplayOneChar(0,0,'3');
}
else
{
h1=1;
h2=0;
if(l1==0 && h1==1 && h2==0 && l2==1 && l3==1 && l4==1 )//6
{
while(l1==0);
}else if(l1==1 && h1==1 && h2==0 && l2==0 && l3==1 && l4==1 )//7
{
while(l2==0);
}else if(l1==1 && h1==1 && h2==0 && l2==1 && l3==0 && l4==1)//8
{
while(l3==0);
}else if(l1==1 && h1==1 && h2==0 && l2==1 && l3==1 && l4==0)//9
{
while(l4==0);
}else
{
h1=1;
h2=1;
h3=0;
if(l1==0 && h1==1 && h2==1 && h3==0 && l2==1 && l3==1 && l4==1)//11
{
while(l1==0);
}else if(l1==1 && h1==1 && h2==1 && h3==0 && l2==0 && l3==1 && l4==1)//12
{
while(l2==0);
}else if(l1==1 && h1==1 && h2==1 && h3==0 && l2==1 && l3==0 && l4==1)//13
{
while(l3==0);
}else if(l1==1 && h1==1 && h2==1 && h3==0 && l2==1 && l3==1 && l4==0 )//14
{
while(l4==0);
}
else
{
h3=1;
h4=0;
if(l1==0 && h1==1 && h2==1 && h3==4 && l2==1 && l3==1 && l4==1)//11
{
while(l1==0);
}else if(l1==1 && h1==1 && h2==1 && h3==4 && l2==0 && l3==1 && l4==1)//12
{
while(l2==0);
}else if(l1==1 && h1==1 && h2==1 && h3==4 && l2==1 && l3==0 && l4==1)//13
{
while(l3==0);
}else if(l1==1 && h1==1 && h2==1 && h3==4 && l2==1 && l3==1 && l4==0 )//14
{
while(l4==0);
}
}
}
}
}
void main()
{
LcdInit();
EA=1; //全局中断开
EX0=1; //外部中断0开() EX1 为外部中断1 (P3^3)
IT0=1; //低电平触发 IT1为中断1
EX1=1; //外部中断0开() EX1 为外部中断1 (P3^3)
IT1=1; //低电平触发 IT1为中断1
/*
DisplayOneChar(0,0,(char)(datt/100+'0'));
DisplayOneChar(1,0,(char)(datt/10%10+'0'));
DisplayOneChar(2,0,(char)(datt%10+'0'));
*/
while(1)
{
key_scan1();
}
}
void Key_INT1( ) interrupt 0 // INT0 1 INT1 2
{
DisplayOneChar(0,0,'A');
}
void Key_INT2( ) interrupt 2 // INT0 1 INT1 2
{
DisplayOneChar(0,0,'B');
}
审核编辑:汤梓红
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。
举报投诉
-
Proteus
+关注
关注
76文章
1688浏览量
105601 -
中断
+关注
关注
5文章
884浏览量
41030 -
矩阵
+关注
关注
0文章
406浏览量
34253 -
按键
+关注
关注
4文章
220浏览量
57344
发布评论请先 登录
相关推荐
扫描矩阵键盘的三种方法:行列扫描法,反转法,状态机法
本帖最后由 默默地努力 于 2013-11-26 15:08 编辑
实现矩阵键盘扫描的三种方法(代码见附件):1.行列扫描法2.反转法3.状态机法(结合定时器中断)第一种和第二
发表于 11-26 15:04
proteus矩阵键盘仿真电路
×4矩阵键盘原理1.逐行扫描2.行列扫描(二)8×8矩阵键盘设计1.proteus电路图设计2.程序设计3.矩阵键盘仿真效果(三)总结(一)
发表于 07-15 08:27
STM32F767+STM32CubeMX I2C通信读写EEPROM数据(采用轮询、DMA、中断三种方式)精选资料推荐
STM32F767+STM32CubeMX I2C通信读写EEPROM数据(采用轮询、DMA、中断三种方式)摘要-前言作为一名STM32的初学者,在学习过程中会遇到很多问题,解决过程中
发表于 08-23 09:08
单片机三种常用的软件架构
一个合适的软件架构不仅结构清晰,而且可以便于开发与维护。便于帮助开发者把握整个工程的框架三种常用的软件架构:1.顺序执行的前后台系统2.时间片轮询系统3.多任务操作系统顺序执行的前后台系统程序以变量
发表于 01-07 08:29
【STM32】4*4矩阵键盘扫描程序(中断方式)
上一篇杂记写了个扫描方式的键盘程序,但是最近在做一个小游戏对CPU响应速度有要求,于是再弄个简单的键盘中断检测程序吧。 总体思路:中断线为4行连接的GPIO口,先初始化矩阵的4行输
发表于 11-26 11:21
•13次下载
Proteus中以中断方式实现矩阵键盘的扫描
前言最近做单片机的课程设计用到矩阵键盘,在此做个记录。1 矩阵键盘的扫描方式使用矩阵键盘时,首先要判断是否有按键按下,这个过程称为
发表于 11-26 12:21
•9次下载
评论