51单片机(郭天祥版)——键盘检测原理及应用实现

实验中我们使用的STC89C52

 


前言

文章内主要概念引自郭天祥老师《新概念51单片机C语言版》一书

主要展示郭天祥老师书中第四章 键盘检测原理及应用实现。分为仿真、实体两部分。

 


一、单片机是什么?

单片机就是在一块硅片上集成了微处理器、存储器及各种输入/输出接口的芯片,这样一块芯片就具有了计算机的属性,因而被成为单片微型计算机,简称单片机。

二、实验步骤

1.独立键盘检测

1.1需求:用数码管的前两位显示一个十进制数,变化范围为00~59,开始时显示00,每按下S2一次,数值加1;每按下S3键一次,数值减1;每按下S4键一次;数值归零;按下S5一次,利用定时器功能使数值开始自动每秒加1,再次按下S5键,数值停止自动加1。

按键检测流程图:

91064df594204323856c10a4fee64407.png

 

1.2代码如下(示例):

#include "reg52.h"
#define uchar unsigned char
#define uint unsigned int 
sbit key1=P3^4;
sbit key2=P3^5;
sbit key3=P3^6;
sbit key4=P3^7;
sbit dula=P2^6;					// 申明U1锁存器的锁存端
sbit wela=P2^7;					// 申明U2锁存器的锁存端
uchar code table[]={  0x3f,0x06,0x5b,0x4f,
					  0x66,0x6d,0x7d,0x07,
					  0x7f,0x6f,0x77,0x7c,
					  0x39,0x5e,0x79,0x71};

void delayms(uint);

uchar numt0,num;

void display(uchar numdis)		// 显示子函数
{
	uchar shi,ge;					// 分离两个分别要显示的数
	shi=numdis/10;
	ge=numdis%10;

	dula=1;
	P0=table[shi];				// 送十位选段数据
	dula=0;
	P0=0xff;					// 送位选数据前关闭所有显示,防止打开位选锁存时
	wela=1;						// 原来段选数据通过位选锁存器造成混乱
	P0=0xfe;
	wela=0;
	delayms(5);				// 延时

	dula=1;
	P0=table[ge];				// 送个位段选数据
	dula=0;
	P0=0xff;
	wela=1;
	P0=0xfd;
	wela=0;
	delayms(5);
}

void delayms(uint xms)
{
	uint i,j;
	for(i=xms;i>0;i--)          	// i=xms即延时约xms毫秒
		for(j=110;j>0;j--);
}

void init()                    // 初始化函数
{
	TMOD=0x01;				// 设置定时器0为工作方式1(0000 0001)
	TH0=(65536-45872)/256;		// 初装值50ms一次中断
	TL0=(65536-45872)%256;
	EA=1;						// 开总中断
	ET0=1;						// 开定时器0中断
}

void keyscan()
{
	if(key1==0)
	{
		delayms(10);
		if(key1==0)
		{
			num++;
			if(num==60)			 // 当到60时重新归0
				num=0;
			while(!key1); 		 // 等待按键释放
		}
	}
	if(key2==0)
	{
		delayms(10);
		if(key2==0)
		{
			if(num==0)			 // 当到0时重新归60
			    num=60;
				num--;
				while(!key2);	
		}
	}
	if(key3==0)
	{
		delayms(10);
		if(key3==0)
		{
			num=0;				 // 清0
			while(!key3);
		}
	}
	if(key4==0)
	{
		delayms(10);
		if(key4==0)
		{
			while(!key4)
			TR0=~TR0;			// 停止或启动定时器0
		}
	}
}

void main()
{
	init();						// 初始化函数
	while(1)
	{
		keyscan();
		display(num);
	}
}

void T0_time() interrupt 1
{
	TH0=(65536-45872)/256;		// 重装初值
	TL0=(65536-45872)%256;
	numt0++;
	if(numt0==20)				// 如果到了20次,说明1秒时间到
	{
		numt0=0;				// 然后把num清0重新再计20次
		num++;
		if(num==60)
			num=0; 
	}
}

1.3图片

TX-1C实验板上独立键盘与单片机连接图:

dbc7ad7b21fa4559bd4550269d901dd6.png


Proteus电路仿真图:

c6e01b3dc68e4163888adc1d4091981a.pngc9c1d6484ee24b7488ac6dc6d862ab4b.png

 本电路图需用到

6位共阴数码管1个;

74HC573锁存器两个;

button按钮4个;

上拉电阻;

电源;

接地

767eb8635e224b9aa9e0f05fe7d0bff7.png

单片机实体图:

ae70aedc6fc8479480c2e19a9cd7cfce.jpeg

8c591c8db91a44e69f4c8b419abb2cf1.jpeg

1.4视频

独立键盘检测

 

 

 

2.矩阵键盘检测

2.1需求:实验板上电时,数码管不显示,顺序按下矩阵键盘后,在数码管上依次显示0~F,6个数码管同时静态显示。

2.2代码如下(示例):

 

#include"reg52.h"
#define uchar unsigned char
#define uint unsigned int 
sbit dula=P2^6;					// 申明U1锁存器的锁存端
sbit wela=P2^7;					// 申明U2锁存器的锁存端
uchar code table[]={  0x3f,0x06,0x5b,0x4f,
					  0x66,0x6d,0x7d,0x07,
					  0x7f,0x6f,0x77,0x7c,
					  0x39,0x5e,0x79,0x71};

void delayms(uint xms)
{
	uint i,j;
	for(i=xms;i>0;i--)			//i=xms即延时约xms毫秒
		for(j=110;j>0;j--);
}

void display(uchar num)
{
	P0=table[num];			//显示函数只送段选数据
	dula=1;
	dula=0;
}

void matrixkeyscan()
{
	uchar temp,key;
	P3=0xfe;
	temp=P3;
	temp=temp&0xf0;
	if(temp!=0xf0)
	{
		delayms(10);
		temp=P3;
		temp=temp&0xf0;
		if(temp!=0xf0)
		{
			temp=P3;
			switch(temp)
			{
				case 0xee:
						key=0;
						break;
				case 0xde:
						key=1;
						break;
				case 0xbe:
						key=2;
						break;
				case 0x7e:
				   		key=3;
						break;
			}
			while(temp!=0xf0)			//等待按键释放
			{
				temp=P3;
				temp=temp&0xf0;
			}
			display(key);		        // 显示
		}
	}
	P3=0xfd;
	temp=P3;
	temp=temp&0xf0;
	if(temp!=0xf0)
	{
		delayms(10);
		temp=P3;
		temp=temp&0xf0;
		if(temp!=0xf0)
		{
			temp=P3;
			switch(temp)
			{
				case 0xed:
						key=4;
						break;
				case 0xdd:
						key=5;
						break;
				case 0xbd:
						key=6;
						break;
				case 0x7d:
						key=7;
						break;
			}
			while(temp!=0xf0)
			{
				temp=P3;
				temp=temp&0xf0;
			}
			display(key);
		}
	}
	P3=0xfb;
	temp=P3;
	temp=temp&0xf0;
	if(temp!=0xf0)
	{
		delayms(10);
		temp=P3;
		temp=temp&0xf0;
		if(temp!=0xf0)
		{
			temp=P3;
			switch(temp)
			{
				case 0xeb:
						key=8;
						break;
				case 0xdb:
						key=9;
						break;
				case 0xbb:
						key=10;
						break;
				case 0x7b:
						key=11;
						break;
			}
			while(temp!=0xf0)
			{
				temp=P3;
				temp=temp&0xf0;
			}
			display(key);
		}
	}
	P3=0xf7;
	temp=P3;
	temp=temp&0xf0;
	if(temp!=0xf0)
	{
		delayms(10);
		temp=P3;
		temp=temp&0xf0;
		if(temp!=0xf0)
		{
			temp=P3;
			switch(temp)
			{
				case 0xe7:
						key=12;
						break;
				case 0xd7:
						key=13;
						break;
				case 0xb7:
						key=14;
						break;
				case 0x77:
						key=15;
						break;
			}
			while(temp!=0xf0)
			{
				temp=P3;
				temp=temp&0xf0;
			}
			display(key);
		}
	}
}

void main()
{
	P0=0;			//关闭所有数码管段选
	dula=1;
	dula=0;
	P0=0xc0;			// 位选中所有数码管
	wela=1;
	wela=0;
    while(1)
	{
		matrixkeyscan();	//不停调用键盘扫描程序
	}

2.3图片

TX-1C实验板上矩阵键盘按键与单片机连接图:

d4c1690820d0446680785e2b485c12cb.png

Proteus仿真电路图:

c6e01b3dc68e4163888adc1d4091981a.png33c1fa800c8647ff951fa48135d31532.png

6位共阴数码管1个;

74HC573锁存器两个;

button按钮4个;

上拉电阻;

电源;

接地

655ef1650fe442f383c509834383a6ee.png

单片机实体:

ae93e4b766864ce4adf9baee412d04b5.jpeg

 

6a94ae8132d6410fa1233bb8870e2cd2.jpeg

acc82e612a4741febf85e726574c29d0.jpeg

2.4视频

矩阵键盘检测

仿真视频:

先用keil编译生成hex文件;

再用Proteus仿真电路选择生成的hex文件进行仿真。

keil编译

Proteus

 

 


 

总结:
以上就是今天要讲的内容,本文仅仅简单介绍了单片机键盘检测的应用实现,而单片机键盘检测相关理论可以参考教材进行学习

83abb35315ffcf77f254754f5e374a36.jpeg

 

  • 18
    点赞
  • 36
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值