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

本文介绍了一种使用TM1638芯片控制16个按键的方法,通过C语言程序实现了按键的读取和应用。包括按键扫描驱动程序的设计与实现,以及按键功能的应用示例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

TM1638驱动程序相关索引

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

一、概述

在中,记录了TM1638显示板的显示功能,本文分享该显示板的按键的读取。
在这里插入图片描述

二、原理图分析

在这里插入图片描述
在这里插入图片描述
如上图,16个按键,每个按键的一端连接TM1638的K1、K2,另一端连接SGn,即KSn。再看手册(手册下载数码管显示板驱动芯片TM1638数据手册中关于按键扫描的部分:
在这里插入图片描述
与8按键8数码管8LED的显示板不同的是,该显示板可以有组合按键,那在写读按键程序的时候,返回的按键可以不是一个独立的按键,所以可以采用一个两个字节的数据,每一位代表一个按键,正好可以存储16个按键的值。

三、主要代码

1. 按键扫描驱动

typedef union
{
	struct
	{
		uint8_t bKey1 	:1;//S1
		uint8_t bKey2 	:1;//S2
		uint8_t bKey3 	:1;//S3
		uint8_t bKey4 	:1;//S4
		uint8_t bKey5 	:1;//S5
		uint8_t bKey6 	:1;//S6
		uint8_t bKey7 	:1;//S7
		uint8_t bKey8 	:1;//S8
		uint8_t bKey9 	:1;//S9
		uint8_t bKey10 	:1;//S10
		uint8_t bKey11	:1;//S11
		uint8_t bKey12	:1;//S12
		uint8_t bKey13	:1;//S13
		uint8_t bKey14	:1;//S14
		uint8_t bKey15	:1;//S15
		uint8_t bKey16	:1;//S16
	}bt;
	struct
	{
		uint8_t lowb;		  
		uint8_t highb;
	}byte;
	uint16_t word;
}Key_tu;
/*******************************************************************************
  * 函数名:TM1638_ReadKey
  * 功  能:TM1638读按键数据
  * 参  数:无
  * 返回值:读出的数据
  * 说  明:返回值为双字节共用体,每一个bit为一个按键;
*******************************************************************************/
uint16_t TM1638_ReadKey(void)
{
	uint8_t u8Data[4], i;
	Key_tu uKey;
	uKey.word = 0;
	TM1638_STBReset();
	TM1638_WriteData(0x42);
	delay_us(5);
	for (i = 0; i < 4; i++)
	{
		u8Data[i] = TM1638_ReadData();//读BYTE1~BYTE4的数据
	}
	TM1638_STBSet();
	
	uKey.bt.bKey1 = (((u8Data[0] & 0x04) == 0x04) ? 1 : 0 );
	uKey.bt.bKey2 = (((u8Data[0] & 0x40) == 0x40) ? 1 : 0 );
	uKey.bt.bKey3 = (((u8Data[1] & 0x04) == 0x04) ? 1 : 0 );
	uKey.bt.bKey4 = (((u8Data[1] & 0x40) == 0x40) ? 1 : 0 );
	uKey.bt.bKey5 = (((u8Data[2] & 0x04) == 0x04) ? 1 : 0 );
	uKey.bt.bKey6 = (((u8Data[2] & 0x40) == 0x40) ? 1 : 0 );
	uKey.bt.bKey7 = (((u8Data[3] & 0x04) == 0x04) ? 1 : 0 );
	uKey.bt.bKey8 = (((u8Data[3] & 0x40) == 0x40) ? 1 : 0 );
	uKey.bt.bKey9 = (((u8Data[0] & 0x02) == 0x02) ? 1 : 0 );
	uKey.bt.bKey10 = (((u8Data[0] & 0x20) == 0x20) ? 1 : 0 );
	uKey.bt.bKey11 = (((u8Data[1] & 0x02) == 0x02) ? 1 : 0 );
	uKey.bt.bKey12 = (((u8Data[1] & 0x20) == 0x20) ? 1 : 0 );
	uKey.bt.bKey13 = (((u8Data[2] & 0x02) == 0x02) ? 1 : 0 );	
	uKey.bt.bKey14 = (((u8Data[2] & 0x20) == 0x20) ? 1 : 0 );
	uKey.bt.bKey15 = (((u8Data[3] & 0x02) == 0x02) ? 1 : 0 );
	uKey.bt.bKey16 = (((u8Data[3] & 0x20) == 0x20) ? 1 : 0 );
	return uKey.word;
}

注意下图中红框圈出的部分,是该显示板连接的按键对应的BIT,理解了写出上面的程序就很简单了。
在这里插入图片描述

2. 按键应用程序

实现与 TM1638驱动显示板(8数码管+8LED+8按键)单片机C语言程序(按键功能)相同的功能,演示效果也相同,因按键读取程序的差异,稍作修改即可:

/*******************************************************************************
  * 函数名:Key_ScanProcess
  * 功  能:按键扫描处理
  * 参  数:无
  * 返回值:无
  * 说  明:去抖动,每20ms扫描一次,扫描时按键被释放才会执行案件的操作;
			仅支持单键、单击
*******************************************************************************/
void Key_ScanProcess(void)
{
	uint8_t u8KeyNum, i;
	u8KeyNum = (uint8_t)(TM1638_ReadKey() & 0x0007);//获取键值
	if (u8KeyNum != 0)//有键按下
	{
		u8KeyState = KEY_PRESSED;
		for (i = 0; i < 3; i++)
		{
			if (((u8KeyNum >> i) & 0x01) == 1)
			{
				u8Key_Name = (i + 1);//保存按键值
				break;
			}
		}
	}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;
	}		
}
#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; } }
### 使用ESP32驱动TM1638按键模块 #### 驱动原理 TM1638是一种集成了LED显示驱动和键盘扫描功能的芯片,广泛应用于数码管显示按键输入场景。通过SPI协议与微控制器通信,可以轻松控制其上的LED点亮以及读取按键状态[^1]。 对于ESP32来说,可以通过GPIO引脚配置来完成对TM1638模块的数据传输、时钟同步以及片选信号的操作。具体而言,STM(数据线)、CLK(时钟线)和DIO(双向I/O线)分别对应到ESP32的指定GPIO引脚上[^2]。 #### 示例代码 以下是基于Arduino IDE编写的用于ESP32驱动TM1638按键模块的示例代码: ```cpp #include <TM1638.h> // 定义 STM, CLK 和 DIO 对应的 GPIO 引脚号 #define STB_PIN 16 // 片选引脚 (STB) #define CLK_PIN 17 // 时钟引脚 (CLK) #define DIO_PIN 25 // 数据引脚 (DIO) // 初始化 TM1638 模块对象 TM1638 module(CLK_PIN, DIO_PIN, STB_PIN); void setup() { Serial.begin(115200); // 开启串口调试 } void loop() { uint8_t key = module.getKey(); // 获取当前按下的键值 if (key != 0) { // 如果检测到有按键按下 Serial.print("Pressed Key: "); Serial.println(key); // 打印按键编号至串口监视器 } delay(100); // 延迟防止频繁读取 } ``` 上述代码实现了基本的功能:初始化TM1638模块并持续监听是否有按键被触发;一旦检测到有效按键事件,则将其对应的数值打印出来[^4]。 #### 库文件安装 为了简化开发流程,在实际项目中通常会借助第三方库 `TM1638` 来快速实现相关功能。该库已经封装好了底层硬件操作细节,开发者只需调用简单的API即可完成复杂任务。可以在Arduino IDE中的“管理库”选项下搜索并安装此库。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值