TM1638驱动显示板(8数码管+8LED+8按键)单片机C语言程序(按键功能)

26 篇文章 1 订阅
19 篇文章 3 订阅

TM1638驱动程序相关索引

1.TM1638显示板(8数码管+8LED+8按键)驱动程序(显示功能)
2.TM1638驱动显示板(8数码管+8LED+8按键)单片机C语言程序(按键功能)
3.TM1638数码管显示板(8数码管+16按键)单片机C语言驱动程序(显示功能)
4.TM1638数码管显示板(8数码管+16按键)单片机C语言驱动程序(按键功能)

一、概述

上一篇文章记录了TM1638驱动的显示模块的显示功能C语言程序,详见TM1638显示板(8数码管+8LED+8按键)驱动程序(显示功能),本文分享按键的驱动,以及按键的去抖动等操作。
模块如下图:
在这里插入图片描述
笔者采用的MCU是STM32F103C8T6,IAR开发环境,使用了HAL库。

二、驱动程序

1. 硬件分析

在这里插入图片描述
首先,通过上图,可以看出按键是连接到K3上的;
在这里插入图片描述
再看上图手册中对按键功能的描述,可知所有按键连接到同一个K引脚上,不能实现组合按键,按键仅能单个使用。读取按键的4个BYTE时,我们需要把K3引脚对应的值取出,即查看每个BYTE的B0和B4位,有一个为1,则对应的键被按下,又因为无法实现组合按键,所以在取4个BYTE中的K3对应的位时,取到第一个1,即认为对应的按键被按下。

2. 按键读取驱动程序

按键读取程序如下:

/*******************************************************************************
  * 函数名:TM1638_ReadKey
  * 功  能:TM1638读按键数据
  * 参  数:无
  * 返回值:读出的数据
  * 说  明:实际该显示板只有8个按键,返回值1~8分别为Key1~Key8;
			所有按键连接K3,无法实现组合按键,只允许一次按一个键;
*******************************************************************************/
uint8_t TM1638_ReadKey(void)
{
	uint8_t u8Data[4], i;
	uint8_t u8Key = 0;
	TM1638_STBReset();
	TM1638_WriteData(0x42);
	for (i = 0; i < 4; i++)
	{
		u8Data[i] = TM1638_ReadData();//读BYTE1~BYTE4的数据
	}
	TM1638_STBSet();
	for (i = 0; i < 4; i++)//取出4个BYTE中的键值
	{
		if (((u8Data[i] >> 0) & 0x01) == 0x01)//第0位,B0,对应按键为1、3、5、7,计算公式是i*2+1
		{
			u8Key = (i * 2 + 1);
			break;
		}else
		{
			if (((u8Data[i] >> 4) & 0x01) == 0x01)//第4位,B4,对应按键为2、4、6、8,计算公式是i*2+2
			{
				u8Key = (i * 2 + 2);
				break;
			}
		}
	}
	return u8Key;
}

程序注释还算详细,应该不需要再说明了。

3.应用层程序

这里我们实现如下功能:
右侧4位数码管显示实际温度,精确到1位小数;左侧4位数码管显示设定温度,精确到1位小数;
左1按键为设置键,按下时,设定温度闪烁显示,左2键为“+”键,可向上调整设定温度,设置步长为0.1,左3键为“-”键,可向下调整设定温度,设置步长为0.1,再次按设置键,温度设定完成,不再闪烁。
根据人的按键操作反应时间,程序可20ms左右执行一次,扫描时按键被释放,才将按键值返回,完成按键相应的操作,达到去抖动的效果。该方法简单易懂,但缺点是没有长按、双击等操作(可自行扩展)。主要代码如下:

#define KEY_NULL								0x00//无
#define KEY_SET									0x01//设置键
#define KEY_UP									0x03//向上键
#define KEY_DOWN								0x05//向下键

typedef union
{
	uint8_t byte;
	struct
	{
		uint8_t bSetMode:		1;//0正常,1进入设定模式
		uint8_t b1:				1;
		uint8_t b2:				1;
		uint8_t b3:				1;
		uint8_t b4:				1;
		uint8_t b5:				1;
		uint8_t b6:				1;
		uint8_t b7:				1;
	}bt;
}KeyFlag_tu;

static uint8_t u8Key_Name;//按键名称
static uint8_t u8KeyState;//按键状态,按下或释放
static KeyFlag_tu uKey_Flag;
static uint32_t u32Key_SettingTemper;//正在设定的温度值

#define KEY_RELEASED					0x00//键被释放
#define KEY_PRESSED						0x01//键被按下
/*******************************************************************************
  * 函数名:Key_Init
  * 功  能:初始化
  * 参  数:无
  * 返回值:无
  * 说  明:无
*******************************************************************************/
void Key_Init(void)
{
	u8Key_Name = KEY_NULL;
	uKey_Flag.byte = 0;
	u8KeyState = KEY_RELEASED;
	u32Key_SettingTemper = 0;
}
/*******************************************************************************
  * 函数名:Key_ScanProcess
  * 功  能:按键扫描处理
  * 参  数:无
  * 返回值:无
  * 说  明:去抖动,每20ms扫描一次,扫描时按键被释放才会执行案件的操作;
			仅支持单键、单击
*******************************************************************************/
void Key_ScanProcess(void)
{
	uint8_t u8KeyNum;
	u8KeyNum = TM1638_ReadKey();//获取键值
	if (u8KeyNum != 0)//有键按下
	{
		u8KeyState = KEY_PRESSED;
		u8Key_Name = u8KeyNum;//保存按键值
	}else
	{
		u8KeyState = KEY_RELEASED;
	}
	if (u8KeyState == KEY_RELEASED)//按键释放
	{
		switch (u8Key_Name)
		{
			case KEY_SET://设置键
			{
				if (uKey_Flag.bt.bSetMode)
				{
					uKey_Flag.bt.bSetMode = 0;//退出设置模式
					PID_ModifySetTemper(u32Key_SettingTemper);//修改设定温度
				}else
				{
					uKey_Flag.bt.bSetMode = 1;//进入设置模式
					u32Key_SettingTemper = PID_GetSetTemper();//读取设定温度
				}			
			}break;
			case KEY_UP://向上键
			{
				if (uKey_Flag.bt.bSetMode)//设置模式
				{
					if (u32Key_SettingTemper < (80 * 100))
					{
						u32Key_SettingTemper += 10;
					}
				}
			}break;
			case KEY_DOWN://向下键
			{
				if (uKey_Flag.bt.bSetMode)//设置模式
				{
					if (u32Key_SettingTemper > (30 * 100))
					{
						u32Key_SettingTemper -= 10;
					}
				}
			}break;
			default:break;			
		}
		u8Key_Name = KEY_NULL;
	}		
}

4.效果

在这里插入图片描述

  • 5
    点赞
  • 38
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
#include //1638he165合并程序 2018/5/26 #include #include #define uchar unsigned char #define uint unsigned int sbit SU0=P2^0; //计数脉冲识别 sbit SU1=P2^1; //计数脉冲识别 sbit SU2=P2^2; sbit DJ1=P1^0; sbit DJ2=P1^1; sbit QH=P3^2; //输出端 sbit CK=P3^3; //时钟 上升沿有效 sbit PL=P3^4; //移位控制 低电平有效 uchar temp; uchar temp1; uchar tempH; uchar tempL; bit weia; bit weib; bit ding; unsigned char num[8]; //各个数码管显示的值 unsigned int dingshiqi=0; unsigned int num1,num2; unsigned int su1_a,su1_b; unsigned int su2_a,su2_b; unsigned int su2,su4,su1; unsigned char wei,wei1; unsigned int k; void delay1ms(uint i) //1ms延时程序 { uchar j; while(i--) { for(j=0;j0; j--); } uint read_int165(void) { uchar i=0; uint read_data=0; PL=0; //置数,读入并行输入数据 _nop_(); PL=1; //移位,并口被锁存,串行转换开始 _nop_(); for(i=0;i<16;i++) //设定16位输入 { read_data<<=1; if(QH) { read_data|=QH; } CK=0; //下降沿 _nop_(); CK=1; _nop_(); //上升沿 } return read_data; } void init_t0() { TMOD = 0x02; //8位自动载定时器 TH0 = 0x06; TL0 = 0x06; TR0 = 1; //启动定时器 ET0=1; //允许定时器0中断 EA=1; //开总中断 } void main(void) { unsigned char i; init_t0(); init_TM1638(); for(i=0;i<8;i++) Write_DATA(i<>8); //获取高八位,存在tempH tempL=(uchar)temp; //获取低八位存在tempL P2=tempH; //接收的字节高八位显示在P2 P1=tempL; //接收的低八位显示在P1 } { i=Read_key(); switch(i) { case 0: //1--1 { while(Read_key()==i); //等待按键释放 su1_a = 0; su2_a = 0; wei=0; weia=1; ding=0; }break; case 1: { while(Read_key()==i); //等待按键释放 weia=0; wei++; if(wei>=3)wei = 0; }break; case 2: { while(Read_key()==i); //等待按键释放 if(wei==1) su1_b++; if(su1_b>5500) su1_b=0; if(wei==2) su2_b++; if(su2_b>5500) su2_b=0; }break; case 3: { while(Read_key()==i); //等待按键释放 if(wei==1) { if(su1_b>0)su1_b--; } if(wei==2) { if(su2_b>0)su2_b--; } }break; case 4: { while(Read_key()==i); //等待按键释放 ding=~ding; }break; case 5:{ while(Read_key()==i); }break; case 6:{ while(Read_key()==i); } break; case 7:{ while(Read_key()==i); }break; } if(wei==0) //脉冲输入计数 { if((ding==0)&&(weia==1)) { if(SU0 ==0) { delay(1); if(SU0==0) { while(!SU0); su1_a++; } } if(SU1 ==0) { delay(1); if(SU1==0) { while(!SU1); su2_a++; } } if(su1_a==su1_b) su1_a = 0; if(su2_a==su2_b) { ding = 1; } } Write_DATA(3*2,tab[su1_a]); Write_DATA(2*2,tab[su1_a0/10]); Write_DATA(1*2,tab[su1_a00/100]); Write_DATA(0*2,tab[su1_a000/1000]); Write_DATA(7*2,tab[su2_a]); Write_DATA(6*2,tab[su2_a0/10]); Write_DATA(5*2,tab[su2_a00/100]); Write_DATA(4*2,tab[su2_a000/1000]); } if(wei==1) //左边数码管设置 { k++; if(k>40)k = 0; if(k>10) { Write_DATA(3*2,tab[su1_b]); Write_DATA(2*2,tab[su1_b0/10]); Write_DATA(1*2,tab[su1_b00/100]); Write_DATA(0*2,tab[su1_b000/1000]); } else { Write_DATA(0*2,tab[20]); Write_DATA(1*2,tab[20]); Write_DATA(2*2,tab[20]); Write_DATA(3*2,tab[20]); } } if(wei==2) //右边数码管设置 { Write_DATA(3*2,tab[su1_a]); Write_DATA(2*2,tab[su1_a0/10]); Write_DATA(1*2,tab[su1_a00/100]); Write_DATA(0*2,tab[su1_a000/1000]); k++; if(k>40)k = 0; if(k>10) { Write_DATA(7*2,tab[su2_b]); Write_DATA(6*2,tab[su2_b0/10]); Write_DATA(5*2,tab[su2_b00/100]); Write_DATA(4*2,tab[su2_b000/1000]); } else { Write_DATA(4*2,tab[20]); Write_DATA(5*2,tab[20]); Write_DATA(6*2,tab[20]); Write_DATA(7*2,tab[20]); } } } } } void time0() interrupt 1 //使用的是定时器T0 { dingshiqi++; if(dingshiqi>=3686) { dingshiqi = 0; } }
好的,这是一个关于嵌入式系统的问题。我会尽力回答,但由于篇幅有限,可能不能详细介绍每一个细节。首先,我们需要了解一些基础知识。 51单片机是一种经典的8位微控制器,具有广泛的应用领域。Keil是一种常用的51单片机开发环境,可以较为方便地进行程序的编写、调试和下载。8×8LED点阵屏是一种比较常见的LED显示屏,可以显示简单的图形和字符。 按键控制8×8LED点阵屏显示图形的过程可以分为以下几个步骤: 1. 硬件连接:将51单片机的I/O口与LED点阵屏的控制芯片连接,同时连接按键单片机的另外一个I/O口。 2. 程序编写:使用Keil开发环境编写程序,包括按键的扫描、LED点阵屏的控制以及图形的显示等。 3. 程序调试:将程序下载到单片机中,通过Keil提供的仿真器进行调试,查看程序是否正常运行。 下面是一个简单的程序框架,可以实现按键控制8×8LED点阵屏显示图形: ``` #include <reg52.h> #define uchar unsigned char #define uint unsigned int sbit key = P1^0; //定义按键连接的I/O口 uchar code table[] = { //定义图形对应的数据 0x18,0x24,0x42,0x81,0x81,0x42,0x24,0x18 }; void delay(uint xms) //延时函数 { uint i,j; for(i=xms;i>0;i--) for(j=110;j>0;j--); } void main() { uchar i,j,k; while(1) { if(key == 0) //检测到按键按下 { for(i=0;i<8;i++) { P2 = ~(1<<i); //选择对应的行 P0 = table[i]; //显示对应的数据 delay(50); //延时一段时间 } } } } ``` 这个程序实现了按键控制8×8LED点阵屏显示一个箭头的图形,具体实现方式为:按下按键后,程序开始循环,每次循环都选择一行,显示对应的数据,并延时一段时间,然后再选择下一行,直到显示完整个图形。在实际应用中,可以根据需要修改程序,实现不同的图形和功能。 希望这个回答能对您有所帮助。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值