STM32 学习笔记]STM32f407 使用 4*4 矩阵键盘
最编程
2024-04-13 22:50:23
...
作者:李剀 出处:https://www.cnblogs.com/kevin-nancy/
欢迎转载,但也请保留上面这段声明。谢谢!
写在前面: 这是本人第一次开始写博客,可能写的不是很好,也请大家谅解。
本人现在大三,以前在学习过程中遇到过各种各样的问题,关于51单片机,STM32单片机,最近在学习ARM11的Tiny6410
以后还会更新一些C/C++方面的东西
关于写博客这件事,其实 一直想写博客记录下来,但是因为某些原因(懒),so , 没有写。现在开始,以后遇到单片机上或者编程上遇到的问题,
并且自己很好地解决了,我就会在这里记录下来。
希望通过博客记录我的学习历程并希望我所解决问题的过程能够帮到需要的人,一起加油吧!!!
我使用的是STM32F407zg的最小系统板
使用HAL库,用STMCubeMX配置
1、首先介绍一下4*4矩阵键盘扫描原理
呃。。。。。就不介绍了,http://www.51hei.com/mcu/3815.html,这个写的挺好的
2、Cube配置
矩阵键盘从下到上依次接到STM32F407zg单片机的PD0~7引脚 ,其中,矩阵键盘下面的四个排针对应的是键盘的行(ROW)
上面四个对应列(COL) 这里我画了一张图来说明我所使用的
PD0-3依次对应3-0行 PD4-7依次对应0-3列 这张图里可以看到我画了两个箭头,下面就给大家说一下我的配置
配置PD0~3为推挽输出 PD4~7为下拉输入,下图是我在STM32CubeMX 5.1.0中配置的
上面的箭头,横向的表示是单片机输出给键盘的,竖向的是表示送给单片机的,也就是单片机配置的输入引脚
用来读取PD4~7的电平
3、程序编写
因为PD4~7为上拉输入,所以是处在一种高阻态(可以暂时理解为高电平,就是四列都为1)
我以扫描第一行为例讲解 ,
上述是软件实现部分, GPIO->IDR和0xf7相与, 若是0xe7 则二进制位1110 0111,说明PD4变为了0,所以第一列的按键
s1按下
note:PD7对应二进制数中的最高位 PD0对应最低位
后面以此类推
注意注意注意:在程序编写完后,可能出现有些行未扫描,有些行的按键按下没有通过串口返回信息
我的解决方法是,在每一行的扫描之前,将行所对应的引脚清空
这样就不会有问题了
4、贴上完整程序:
#include "keypad.h"
#include "stdint.h"
#include "stm32f4xx_hal.h"
uint16_t Key_scan(void)
{
uint16_t Key_val = 0; // 按键扫描返回键值,初始化为0
// 强调一下: 这里必须付一个初值0 否则串口打印出错
uint16_t temp;
/*=========================以下代码是按键扫描程序=========================*/
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_0|GPIO_PIN_1|GPIO_PIN_2|GPIO_PIN_3, GPIO_PIN_RESET); // 先清空引脚状态
/*----------------------------Scan the 1st ROW----------------------------*/
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_0|GPIO_PIN_1|GPIO_PIN_2, GPIO_PIN_SET); // 设置PD0~2为1
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_3, GPIO_PIN_RESET); // 设置PD3为0 二进制编码为 0111
if((GPIOD->IDR & 0xF0) != 0xF0)
{
HAL_Delay(10); // 10ms延时消抖
if((GPIOD->IDR & 0xF0) != 0xF0)
{
temp = (GPIOD->IDR & 0xF7); // GPIOD->IDR寄存器为端口输入数据寄存器
switch(temp) // 用来读取GPIO口的电平状态
{
case 0xE7 : Key_val = 1;
break;
case 0xD7 : Key_val = 2;
break;
case 0xB7 : Key_val = 3;
break;
case 0x77 : Key_val = 4;
break;
default : Key_val = 0; break;
}
}
}
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_0|GPIO_PIN_1|GPIO_PIN_2|GPIO_PIN_3, GPIO_PIN_RESET);
/*----------------------------Scan the 2nd ROW----------------------------*/
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_0|GPIO_PIN_1|GPIO_PIN_3, GPIO_PIN_SET);
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_2, GPIO_PIN_RESET);
if((GPIOD->IDR & 0xF0) != 0xF0)
{
HAL_Delay(10); // 10ms延时消抖
if((GPIOD->IDR & 0xF0) != 0xF0)
{
temp = (GPIOD->IDR & 0xFB);
switch(temp)
{
case 0xEB : Key_val = 5;
break;
case 0xDB : Key_val = 6;
break;
case 0xBB : Key_val = 7;
break;
case 0x7B : Key_val = 8;
break;
default : Key_val = 0; break;
}
}
}
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_0|GPIO_PIN_1|GPIO_PIN_2|GPIO_PIN_3, GPIO_PIN_RESET);
/*----------------------------Scan the 3rd ROW----------------------------*/
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_0|GPIO_PIN_2|GPIO_PIN_3, GPIO_PIN_SET);
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_1, GPIO_PIN_RESET);
if((GPIOD->IDR & 0xF0) != 0xF0)
{
HAL_Delay(10); // 10ms延时消抖
if((GPIOD->IDR & 0xF0) != 0xF0)
{
temp = (GPIOD->IDR & 0xFD);
switch(temp)
{
case 0xED : Key_val = 9;
break;
case 0xDD : Key_val = 10;
break;
case 0xBD : Key_val = 11;
break;
case 0x7D : Key_val = 12;
break;
default : Key_val = 0; break;
}
}
}
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_0|GPIO_PIN_1|GPIO_PIN_2|GPIO_PIN_3, GPIO_PIN_RESET);
/*----------------------------Scan the 4th ROW----------------------------*/
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_1|GPIO_PIN_2|GPIO_PIN_3, GPIO_PIN_SET);
HAL_GPIO_WritePin(GPIOD, GPIO_PIN_0, GPIO_PIN_RESET);
if((GPIOD->IDR & 0xF0) != 0xF0)
{
HAL_Delay(10); // 10ms延时消抖
if((GPIOD->IDR & 0xF0) != 0xF0)
{
temp = (GPIOD->IDR & 0xFE);
switch(temp)
{
case 0xEE : Key_val = 13;
break;
case 0xDE : Key_val = 14;
break;
case 0xBE : Key_val = 15;
break;
case 0X7E : Key_val = 16;
break;
default : Key_val = 0; break;
}
}
}
return Key_val;
}
5、 这是我的测试结果,见下图:
这个程序中的编码只是对应于我自己的接线,自己写的时候要注意自己的接线
上面有些连接是我上传的图片,我自己看的时候看不到图片,不知道什么原因,但是复制链接打开就是图片了
let‘s dream high 每个不满意的现在,都有一个不努力的曾经。加油吧
推荐阅读
-
STM32 学习与实践笔记 (4):分析和理解 GPIO_InitTypeDef GPIO_InitStructure (a)
-
NeurIPS 2022 | 最强斗地主AI!网易互娱AI Lab提出基于完美信息蒸馏的方法-完美信息蒸馏(PTIE) 在斗地主游戏中,非完美信息的引入主要是由于三位玩家均不能看到别人的手牌,对于任意一位玩家而言,仅可知道其余两位玩家当前手牌的并集,而难于精准判断每位玩家当前手牌。完美信息蒸馏的思路是针对这种非完美问题,构建一个第三方角色,该角色可以看到三位玩家的手牌,该角色在不告知每位玩家完美信息的情况下通过信息蒸馏的方式引导玩家打出当前情况下合理的出牌。 以强化学习常用的 Actor-Critic 算法为例,PTIE 在 Actor-Critic 算法的应用中可以利用 Critic 的 Value 输出作为蒸馏手段来提升 Actor 的表现。具体而言即在训练中 Critic 的输入为完美信息(包含所有玩家的手牌信息),Actor 的输入为非完美信息(仅包含自己手牌信息),此种情况下 Critic 给予的 Value 值包含了完美信息,可以更好地帮助 Actor 学习到更好的策略。 从更新公式上来看,正常的 Actor-Critic 算法 Actor 更新的方式如下: 在 PTIE 模式下,对于每个非完美信息状态 h,我们可以在 Critic 中构建对应的完美信息状态 D(h),并用 Critic 的输出来更新 Actor 的策略梯度,从而达到完美信息蒸馏的效果。 PTIE 框架的整体结构如下图所示: 无论是训练还是执行过程中智能体都不会直接使用完美信息,在训练中通过蒸馏将完美信息用于提升策略,从而帮助智能体达到一个更高的强度。 PTIE 的另一种蒸馏方式是将完美信息奖励引入到奖励值函数的训练中,PerfectDou 提出了基于阵营设计的完美信息奖励 node reward,以引导智能体学习到斗地主游戏中的合作策略,其定义如下: 如上所示,完美信息部分 代表 t 时刻地主手牌最少几步可以出完,在斗地主游戏中可以近似理解为是距游戏获胜的距离, 代表 t 时刻地主阵营和农民阵营距游戏获胜的距离之差, 为调节系数。通过此种奖励设计,在训练时既可以一定程度地引入各玩家的手牌信息(出完的步数需要知道具体手牌才能计算),同时也鼓励农民以阵营的角度做出决策,提升农民的合作性。 特征构建: PerfectDou 针对牌类游戏的特点主要构建了两部分特征:牌局状态特征和动作特征。其中牌局状态特征主要包括当前玩家手牌牌型特征、当前玩家打出的卡牌牌型特征、玩家角色、玩家手牌数目等常用特征,动作特征主要用于刻画当前状态下玩家的所有可能出牌,包括了每种出牌动作的牌型特征、动作的卡牌数目、是否为最大动作等特征。 牌型特征为 12 * 15 的矩阵,如下图所示: 该矩阵前 4 行代表对应每种卡牌的张数,5-12 行代表该种卡牌的种类和对应位置。 网络结构和动作空间设计 针对斗地主游戏出牌组合数较多的问题,PerfectDou 基于 RLCard 的工作上对动作空间进行了简化,对占比最大的两个出牌牌型:飞机带翅膀和四带二进行了动作压缩,将整体动作空间由 27472 种缩减到 621 种。 PerfectDou 策略网络结构如下图所示: 策略网络结构同样分为两部分:状态特征部分和动作特征部分。 在状态特征部分,LSTM 网络用于提取玩家的历史行为特征,当前牌局状态特征和提取后的行为特征会再通过多层的 MLP 网络输出当前的状态信息 embedding。 在动作特征部分,每个可行动作同样会经过多层 MLP 网络进行编码,编码后的动作特征会与其对应的状态信息 embedding 经过一层 MLP 网络计算两者间的相似度,并经由 softmax 函数输出对应的动作概率。 实验结果
-
STM32 4*4 矩阵键盘实现原理(附程序)
-
stm32 矩阵键盘学习笔记
-
STM32 4*4 矩阵键盘实现原理(可对任意引脚进行编程)
-
STM32 4*4 矩阵键盘快速入门使用及其实现 编程 HAL 库/标准库
-
STM32 实现 4*4 矩阵键盘扫描(HAL 库、标准库等)
-
STM32 扫描 4x4 矩阵键盘模块】 4x4 矩阵键盘接口4x4 矩阵键盘接口
-
使用基于 STM32F407 的 4*4 矩阵键盘(含完整程序)
-
STM32 学习笔记]STM32f407 使用 4*4 矩阵键盘