嵌入式软件--51单片机 DAY 3

news/2024/9/19 18:36:49/ 标签: 51单片机, 嵌入式硬件, 单片机

一、独立按键

按键的作用相当于一个开关,按下时接通(或断开),松开后断开(或接通)。

(1)需求

通过SW1、SW2、SW3、SW4四个独立按键分别控制LED1、LED2、LED3、LED4的亮灭,具体要求是,按一下(按下并松开)SW,LED点亮,再按一下SW,LED熄灭。

(2)硬件设计

<1>思路

为实现上述需求,需要设法令单片机感知到按键被按下,也就是说在按键被按下时,需要向单片机发送一个信号,当单片机收到该信号后,再执行控制LED的逻辑即可。

由于单片机>51单片机的GPIO引脚的默认均为高电平,因此只需将按键的一侧接入单片机的某个GPIO引脚,另一侧接地。这样一来,当按键按下时,引脚直接接地,就相当于向单片机发送了一个低电平信号。

<2>原理图

SW1  P42

SW2  P43

SW3  P32

SW4  P33

<3>软件设计

按照原理图,SW1按下时,P4.2引脚会被拉低;SW2按下时,P4.3引脚会被拉低;SW3被按下时,P3.2引脚会被拉低;SW4按下时,P3.3引脚会被拉低。

因此只需检测上述引脚是否变为低电平即可,若检测到变为低电平,就执行控制LED的逻辑。需要注意的是,按键的检测需要持续进行,所以需要不停的检查上述引脚是否变为低电平。

P42=0,SW1被按下,LED1亮 P00=0,再按下SW1,P00=1,LED1灭。独立按键一直实现低电平才能控制亮灭。

#include <STC89C5xRC.H>
#define SW1  P42
#define SW2  P43
#define SW3  P32
#define SW4  P33
#define LED1 P00
#define LED2 P01
#define LED3 P02
#define LED4 P03void main()
{while (1) { // 持续检测按键if (SW1 == 0) {LED1 = ~LED1;}if (SW2 == 0) {LED2 = ~LED2;}if (SW3 == 0) {LED3 = ~LED3;}if (SW4 == 0) {LED4 = ~LED4;}}
}

这个有缺陷,出现上述现象的原因是,当前代码的逻辑存在问题。当前代码的逻辑是,当按键处在按下的状态时,控制LED的逻辑会被多次触发,也就是LED会快速的闪烁,由于速度过快,人眼不能分辨出来,所以看起来LED就常亮的,但是亮度要比正常情况下低。当按键抬起后,控制LED的逻辑便不会再触发,此时LED会随机的处在按键抬起前最后一刻的状态。

要解决上述问题,就需要确保按键被按下时,控制LED的逻辑只被触发一次。

按键单次触发:

当前代码的逻辑是,只要按键处在按下的状态,就触发控制LED的逻辑。为实现单次触发,可以在按键按下后,等待按键抬起,并在抬起的一刻,执行控制LED的逻辑。

#include <STC89C5xRC.H>
#define SW1  P42
#define SW2  P43
#define SW3  P32
#define SW4  P33
#define LED1 P00
#define LED2 P01
#define LED3 P02
#define LED4 P03void main()
{while (1) { // 持续检测按键if (SW1 == 0) {while (SW1 == 0); // 等待按键抬起LED1 = ~LED1;}if (SW2 == 0) {while (SW2 == 0);LED2 = ~LED2;}if (SW3 == 0) {while (SW3 == 0);LED3 = ~LED3;}if (SW4 == 0) {while (SW4 == 0);LED4 = ~LED4;}}
}

上述代码已经能够实现预期效果了,但是偶尔会“失灵”。这个现象是由按键的物理特性导致的,由于按键内部的弹簧和触点具有弹性,当按键被按下或释放时,这些弹性材料会震动,从而导致触点在短时间内反复接触和断开。所以表面上我们虽然只按了一次按键,但实际上是按了多次。

了解了按键抖动的问题,也就能够解释当前的现象了。如果实际按下的次数是奇数,那么LED就会按照预期点亮或熄灭;而如果是偶数,LED最终还是会回到按下按键之前的状态,看起来就像是按键“失灵”了。

按键消抖

所以,当我们检测到信号为0时,不能莽撞的认为按键已经被按下了,而是要稍微等一段时间(约10ms),等按键稳定之后,再次检测,如果信号仍然为0,才可以确定按键确实被按下了。具体代码如下。

#include <STC89C5xRC.H>
#include "Com_Util.h"
#define SW1 P42
#define SW2 P43
#define SW3 P32
#define SW4 P33
#define LED1 P00
#define LED2 P01
#define LED3 P02
#define LED4 P03
void main()
{while (1){if (SW1==0){Com_Util_Delay1ms(10);if(SW1==0){while (SW1==0);LED1=~LED1;}}if (SW2==0){Com_Util_Delay1ms(10);if(SW2==0){while (SW2==0);LED2=~LED2;}}if (SW3==0){Com_Util_Delay1ms(10);if(SW3==0){while (SW3==0);LED3=~LED3;}}if (SW4==0){Com_Util_Delay1ms(10);if(SW4==0){while (SW4==0);LED4=~LED4;}}}while (1) {}
}

<4>规范代码 

我们将检测按键的代码单独抽取到Int层

Int_Key.c

#include "Int_Key.h"
#define SW1 P42
#define SW2 P43
#define SW3 P32
#define SW4 P33bit Int_Key_IsSW1Pressed()
{if (SW1 == 0) {Com_Util_Delay1ms(10); // 延时消抖if (SW1 == 0) {return 1;}}return 0;
}bit Int_Key_IsSW2Pressed()
{if (SW2 == 0) {Com_Util_Delay1ms(10); // 延时消抖if (SW2 == 0) {while (SW2 == 0); // 等待按键抬起return 1;}}return 0;
}bit Int_Key_IsSW3Pressed()
{if (SW3 == 0) {Com_Util_Delay1ms(10); // 延时消抖if (SW3 == 0) {return 1;}}return 0;
}bit Int_Key_IsSW4Pressed()
{if (SW4 == 0) {Com_Util_Delay1ms(10); // 延时消抖if (SW4 == 0) {while (SW4 == 0); // 等待按键抬起return 1;}}return 0;
}

Int_Key.h

#ifndef __INT_KEY_H__
#define __INT_KEY_H__#include <STC89C5xRC.H>
#include "Com_Util.h"
//检查按键SW1是否放下
//bit是否
bit Int_Key_IsSW1Pressed();bit Int_Key_IsSW2Pressed();bit Int_Key_IsSW3Pressed();bit Int_Key_IsSW4Pressed();
#endif /* __INT_KEY_H__ */

main.c

#include "Int_Key.h"
#define LED1 P00
#define LED2 P01
#define LED3 P02
#define LED4 P03
void main()
{while (1) {if (Int_Key_IsSW1Pressed()) {LED1 = ~LED1;}if (Int_Key_IsSW2Pressed()) {LED2 = ~LED2;}if (Int_Key_IsSW3Pressed()) {LED3 = ~LED3;}if (Int_Key_IsSW4Pressed()) {LED4 = ~LED4;}}
}

二、矩阵按键

1.需求

按下按键矩阵中的SW5到SW20按键后,数码管显示对应的按键编号。

2.硬件设计

由于按键矩阵中共有4x4=16个按键,如果每个按键都接入一个GPIO引脚,势必会造成引脚的浪费,为了节省引脚,我们同样可以借用动态扫描的思想,具体逻辑如下。

扫描流程:

将第一行进行检测,置为0接地。只要p24、p25、p26、p27.谁变成低电平,就是那个被按了。

故,第二行也是如此。这四行只能有一行接地,才能知道哪个按了。之后三四行依次接地。

选中哪一行,哪一行的引脚都会成为低电平,按下改行的按钮,该列的引脚电平会拉低。

行:P20 P21 P22 P23

列:P24 P25 P26 P27 默认都是高电平,只有按下按钮导通才会拉低电平。

3.软件设计

需求:按下SW5--SW20,数码管显示对应的编号。

检查是否被按下,函数返回编号,编号被数码管函数调用刷新显示。

Int_KeyMatrix.h

#ifndef __INT_KEYMATRIX_H__
#define __INT_KEYMATRIX_H__
#include "Com_Util.h"
//检测按键是否被按下,如果被按下就返回,否则返回0
u8 Int_KeyMatrix_CheckSW();#endif /* __INT_KEYMATRIX_H__ */

Int_KeyMatrix.c

#include "Int_KeyMatrix.h"
#include <STC89C5xRC.H>
u8 Int_KeyMatrix_CheckSW()
{//检测第一行P2=0XFE;//1111 1110if (P24==0){Com_Util_Delay1ms(10);if(P24==0){while (P24=0);return 5; }}if (P25==0){Com_Util_Delay1ms(10);if(P25==0){while (P25=0);return 6; }}if (P26==0){Com_Util_Delay1ms(10);if(P26==0){while (P26=0);return 7; }}if (P27==0){Com_Util_Delay1ms(10);if(P27==0){while (P27=0);return 8; }}//检测第二行P2=0xFD;   //1111 1101if (P24==0){Com_Util_Delay1ms(10);if(P24==0){while (P24=0);return 9; }}if (P25==0){Com_Util_Delay1ms(10);if(P25==0){while (P25=0);return 10; }}if (P26==0){Com_Util_Delay1ms(10);if(P26==0){while (P26=0);return 11; }}if (P27==0){Com_Util_Delay1ms(10);if(P27==0){while (P27=0);return 12; }}//检测第三行P2=0XFB;  //1111 1011if (P24==0){Com_Util_Delay1ms(10);if(P24==0){while (P24=0);return 13; }}if (P25==0){Com_Util_Delay1ms(10);if(P25==0){while (P25=0);return 14; }}if (P26==0){Com_Util_Delay1ms(10);if(P26==0){while (P26=0);return 15; }}if (P27==0){Com_Util_Delay1ms(10);if(P27==0){while (P27=0);return 16; }}//检测第四行P2=0XF7; //1111 0111if (P24==0){Com_Util_Delay1ms(10);if(P24==0){while (P24=0);return 17; }}if (P25==0){Com_Util_Delay1ms(10);if(P25==0){while (P25=0);return 18; }}if (P26==0){Com_Util_Delay1ms(10);if(P26==0){while (P26=0);return 19; }}if (P27==0){Com_Util_Delay1ms(10);if(P27==0){while (P27=0);return 20; }
}
}

Int_DigitalTube.h

#ifndef __INT_DIGITALTUBE_H__
#define __INT_DIGITALTUBE_H__
#define SMG_EN P36
#define LED_EN P34#include "Com_Util.h"void Int_DigitalTube_Init();
void Int_DigitalTube_DisplayNum(u32 num);
void Int_DigitalTube_Refresh();#endif /* __INT_DIGITALTUBE_H__ */

Int_DigitalTube.c

#include "Int_DigitalTube.h"
#include <STC89C5xRC.H>// 数字0-9的编码
static u8 s_codes[10] = {0x3F, // 00x06, // 10x5B, // 20x4F, // 30x66, // 40x6D, // 50x7D, // 60x07, // 70x7F, // 80x6F  // 9
};static u8 s_buffer[8];void Int_DigitalTube_Init()
{SMG_EN = 0;LED_EN = 0;
}/*** @brief 内部方法,让数码管某一位显示特定数字** @param position 片选, 从左到右[0-7]* @param num_code 显示想要的数字编码*/
static void Int_DigitalTube_DisplaySingle(u8 position, u8 num_code)
{P0 = 0x00;// 位选:P15 P14 P13position <<= 3;P1 &= 0xC7;P1 |= position;// 段选:P0P0 = num_code;
}void Int_DigitalTube_DisplayNum(u32 num)
{u8 i;for (i = 0; i < 8; i++) {s_buffer[i] = 0x00;}if (num == 0) {s_buffer[7] = s_codes[0];return;}i = 7;while (num > 0) {s_buffer[i] = s_codes[num % 10];num /= 10;i--;}
}void Int_DigitalTube_Refresh()
{u8 i;for (i = 0; i < 8; i++) {Int_DigitalTube_DisplaySingle(i, s_buffer[i]);Com_Util_Delay1ms(1);}
}

main.c

#include "Int_KeyMatrix.h"
#include "Int_DigitalTube.h"
void main()
{u8 key;Int_DigitalTube_Init();while (1){       key=Int_KeyMatrix_CheckSW();if (key){Int_DigitalTube_DisplayNum(key);}Int_DigitalTube_Refresh();}}

4.按键阻塞问题

按下按键不松手的时候,扫描停了,数码管数字只显示最后一位,而且非常亮。

阻塞到了检测步骤,所以最后一个亮,但不扫描了,所以这样了。

5.改造矩阵按键代码

只需改造Int_KeyMatrix.c

#include "Int_KeyMatrix.h"
#include <STC89C5xRC.H>
u8 Int_KeyMatrix_CheckSW()
{u8 i;u8 lines[4]={0xFE,0XFD,0XFB,0XF7};for ( i = 0; i < 4; i++){P2=lines[i];if (P24==0){Com_Util_Delay1ms(10);if(P24==0){while (P24=0);return 5+4*i; }}if (P25==0){Com_Util_Delay1ms(10);if(P25==0){while (P25=0);return 6+4*i; }}if (P26==0){Com_Util_Delay1ms(10);if(P26==0){while (P26=0);return 7+4*i; }}if (P27==0){Com_Util_Delay1ms(10);if(P27==0){while (P27=0);return 8+4*i; }}}
}

把代表行的值写入数组,for循环遍历数组,循环每一行,找到i与列数值上的变化规律。

6.作业1:数码管分四组,独立按键分别与其计数


http://www.ppmy.cn/news/1524024.html

相关文章

在Unity环境中使用UTF-8编码

为什么要讨论这个问题 为了避免乱码和更好的跨平台 我刚开始开发时是使用VS开发,Unity自身默认使用UTF-8 without BOM格式,但是在Unity中创建一个脚本,使用VS打开,VS自身默认使用GB2312(它应该是对应了你电脑的window版本默认选取了国标编码,或者是因为一些其他的原因)读取脚本…

如何评价BEV seg模型的指标

1 可视化 seg PRED GT # 传进来的时没经过sigmoid激活的[1,1,200,200]的featuremap def seg_vis(pred,name):# ---- 区分前景,背景

Python利用pyecharts实现数据可视化

小编会持续更新知识笔记&#xff0c;如果感兴趣可以三连支持。闲来无事&#xff0c;水文一篇&#xff0c;不过上手实践一下倒还是挺好玩的&#xff0c;这一块知识说不定以后真可以尝试拿来做数据库的报表显示。 有梦别怕苦&#xff0c;想赢别喊累。 目录 前言 JSON数据格式的…

风中摇曳的小萝卜(机器学习)笔记 支持向量机

支持向量机就是找到一条直线&#xff0c;让两边的点与它的距离是最大的 只想两边的点为y1和y-1 然后就可以列出到直线的距离了 看分子的正负就可以区分点在直线的哪一边了 上面的表示为函数间隔 下面表示为函数范数 然后我们找到间隔最小的那些点&#xff0c;让最终直线距离…

# 键盘字母上有下标数字,输入时怎么一键去掉,关闭键盘上的下标数字。‌

键盘字母上有下标数字&#xff0c;输入时怎么一键去掉&#xff0c;关闭键盘上的下标数字。‌ 一、问题描述&#xff1a; 如下图&#xff0c;有的笔记本电脑键盘上&#xff0c;没有数字小键盘&#xff0c;数字小键盘会和字母混和在一起&#xff0c;这样打字时&#xff0c;不容…

期权不同手续费怎么收取?期权手续费要交多少?

今天期权懂带你了解期权不同手续费怎么收取&#xff1f;期权手续费要交多少&#xff1f;期权手续费是指在进行期权交易时需要支付的费用&#xff0c;这些费用通常包括交易佣金、交易所费用等等&#xff0c;期权手续费按张数收费。 50ETF期权交易所手续费要交多少&#xff1f; …

一文说清什么是数据仓库

01 数据仓库的概念 数据仓库的概念可以追溯到20世纪80年代&#xff0c;当时IBM的研究人员开发出了“商业数据仓库”。本质上&#xff0c;数据仓库试图提供一种从操作型系统到决策支持环境的数据流架构模型。 目前对数据仓库&#xff08;Data Warehouse&#xff09;的标准定义&a…

【MATLAB】模拟退火算法

模拟退火算法的MATLAB实现 模拟退火算法简介模拟退火算法应用实例关于计算结果 模拟退火算法简介 1982年&#xff0c;Kirkpatrick 将退火思想引入组合优化领域&#xff0c;提出了一种能够有效解决大规模组合优化问题的算法&#xff0c;尤其对 NP 完全问题表现出显著优势。模拟…

跨部门SOP与统一知识库:打破信息孤岛,促进团队协作

引言&#xff1a; 在当今这个快速变化且高度竞争的商业环境中&#xff0c;企业面临着前所未有的挑战&#xff0c;其中之一便是如何高效地跨越部门界限&#xff0c;实现无缝协作。传统的组织结构往往导致信息孤岛的出现&#xff0c;不同部门间流程不一致、信息不共享&#xff0…

写在OceanBase开源三周年

我收获的深刻感触get 感触1&#xff1a;解决问题才有生存价值 [产品力] 感触2&#xff1a;永无止境的“易用性” [易用性] 感触3&#xff1a;立下“双赢”的flag 感触4&#xff1a;社区建设离不开用户和开发者参与 感触5&#xff1a;从易用到用户自助 [自助能力] 当时想法很简…

ARM 全局变量更换基址寄存器

思索了两个晚上&#xff0c;一直没有弄清几万条指令里唯一的 [x27, #80] 来自哪里&#xff0c;结合上下文判定是个全局变量&#xff0c;通过动态调试给main传参数&#xff0c;确认参数 s 时会改变 [x27, #80] 的值&#xff0c;由 0 变 1。于是反推到 case s 处&#xff0c;发现…

leaflet【十】实时增加轨迹点轨迹回放效果实现

实时轨迹回放 在前面有用leaflet-trackplayer实现了一个轨迹回放的效果&#xff0c;单击前往&#xff1a;轨迹回放效果&控制台控制轨迹运动效果 这篇文章主要是实现一下实时增加轨迹点&#xff0c;不改变原来运行轨迹和速度。这里是简易做了一个demo效果&#xff0c;大概…

想要从OPPO手机恢复数据?免费OPPO照片视频恢复软件

此实用程序可帮助那些寻找以下内容的用户&#xff1a; 在OPPO手机中格式化存储卡后可以恢复图片吗&#xff1f;我删除了 OPPO上的视频和图片&#xff0c;我感觉很糟糕&#xff0c;因为里面有我在拉斯维加斯拍摄的视频和照片 免费OPPO照片视频恢复软件 您能恢复OPPO上已删除的…

漫谈设计模式 [14]:迭代器模式

引导性开场 菜鸟: 最近在处理一个项目时&#xff0c;我遇到一个问题。我需要遍历一个集合中的元素&#xff0c;但集合的结构可能会变化。每次我都要写很多重复的代码&#xff0c;这让我很困惑。有没有更好的方法呢&#xff1f; 老鸟: 这个问题很常见。你有没有听说过迭代器…

OpenCV结构分析与形状描述符(17)判断轮廓是否为凸多边形的函数isContourConvex()的使用

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 测试轮廓的凸性。 该函数测试输入的轮廓是否为凸的。轮廓必须是简单的&#xff0c;即没有自相交。否则&#xff0c;函数的输出是不确定的。 cv:…

Java 学习中使用文件、网络连接等资源时,未正确关闭资源,导致资源泄漏应该怎么办?

在Java编程中&#xff0c;处理文件、网络连接、数据库连接等资源时&#xff0c;如果没有正确关闭资源&#xff0c;就会发生资源泄漏。资源泄漏会导致系统性能下降、内存占用增加&#xff0c;甚至可能导致程序崩溃&#xff0c;特别是在高负载的系统中。 一、什么是资源泄漏&…

flink的大状态复用

在 Apache Flink 中&#xff0c;实现大状态复用主要涉及在不同任务、不同生命周期阶段&#xff0c;甚至不同作业之间共享或重用状态数据。复用大状态可以减少重新加载和重新计算的开销&#xff0c;从而提升作业的效率和业务连续性。下面是几种在 Flink 中复用大状态的常见方法&…

Ubuntu 比较两个文件夹

比较两个文件夹下的大量文件是否一致&#xff0c;可以通过以下几种方式完成&#xff1a; 1. 使用 diff 命令 diff 命令不仅可以比较文件&#xff0c;还能递归比较文件夹。可以使用 -r 选项来递归比较两个目录下的文件&#xff1a; diff -r /path/to/dir1 /path/to/dir2 如…

OpengGL教程(三)---使用VAO和VBO方式绘制三角形

本章参考官方教程&#xff1a;learnopengl-cn VertexShader.glsl #version 330 core layout(location 0) in vec3 position; layout(location 1) in vec3 color; uniform mat4 projection; // 投影矩阵 out vec4 ourColor; void main() {gl_Position projection * vec4(p…

Oracle使用序列后提示违反唯一约束---解决办法

1、问题原因分析 出现这个问题的原因是插入数据的时候&#xff0c;由于之前没有使用序列插入&#xff0c;而是直接插入了一个比当前序列nextval还大的值&#xff0c;即直接将id写死了。后面再使用序列插入的时候&#xff0c;如果序列小于该值的话&#xff0c;是可以正常插入的…