51单片机应用篇-- --数码管60秒计时,独立按键可调

2023-05-16

开篇先说一句废话····
本旺名字叫萨摩耶,,Please 叫我旺财,,,哈哈,招财进宝嘛!

缘由

本来按照我的学习计划,我现在应该是单片机的学习过程,应该写单片机学习篇,但是因为在10月中旬收到一则私信,说让我帮忙写一个用独立按键控制数码管的小程序,我看到消息之后,构思了一下,我觉得这里应该会回到定时器中断来控制秒数,所以先去学习中断然后去写,不断的调试,弄好之后,最近学模数数模转换模块有点难,我觉得先缓一下,然后想起这事,特来写一篇文章总结一下。

项目需求

直接上图–
在这里插入图片描述

项目分析

其实上面的需求也算大概分析啦,细致的分析一下。

  • 首先得任务就是数码管循环显示00-59,如果没有每隔一秒的话,可以设置一个for循环来控制数码管显示时间,也可以用DeBug功能来调试出大概一秒,但是这些不怎么准确,所以要使用定时器中断来控制一秒。而00-59的值则设置一个全局变量num来存放。数码管显示的时候,直接设置十位和个位对num分别求模和求余,然后在数码管上动态显示即可。
  • 接着就是独立按键,这里我用的是类似于矩阵按键的检测方法,首先给一个高电平,然后根据按键后返回的值来判断哪个键按下,因为这里有四个按键,写在一起容易搞混,直接写四个函数,按下按键直接进入对应函数处理,因为K0可能会按到两次,对应K1和K2有不同的功能,那么可以设置一个标志位在K1和K2函数里判别K0按下几次,选择对应的功能。
  • 按下K0之后,进入K0函数处理,设置一个全局变量flag,首先默认为0,按下一下为1,再次按下之后,为2,而按下K0之后要调整秒数,那么数码管就得暂停,也就是定时器暂停(TR0=0;)
  • 按下K1和K2之后,首先先判断标志位flag的值,如果为0,也就是没有按过K0,那么直接跳出函数就好,如果为1,也就是K0按下一次,可以对个位处理,K1为加,即加1,K2为减,即减1,要注意的是,当num为00或者59是减1或者加1必须处理,因为00–59,不会出现超过59的数。如果为2,也就是K0按下两次,可以对十位处理,K1为加,即加10,K2为减,即减10,要注意的是,当num为00或者59是减10或者加10必须处理,因为00–59,不会出现超过59的数。
  • 按下K3之后,表示按键完成,数码管接着显示秒数,即定时器启动(TR0=1;)
    主函数里面不断的按键检测然后数码管显示即可。

项目代码

  • 数码管循环显示00–59
//数码管显示函数
void Showsmg(u8 shi,u8 ge){       //十位  个位
	//十位
	LSA=1;LSB=1;LSC=1;
	P0=smgduan[shi];
	Delay(1);
	P0=0x00;
	//个位
	LSA=0;LSB=1;LSC=1;
	P0=smgduan[ge];
	Delay(1);
	P0=0x00;
}
//延时函数
void Delay(u16 i)   //11.0592 1ms
{
	while(i--){
		unsigned char i, j;

		_nop_();
		i = 2;
		j = 199;
		do
		{
			while (--j);
		} while (--i);
	}
} 
//初始化定时器函数
void Init(){
	t0=0;
	num=0;
	shi=0;
	ge=0;
	flag=0;
	TMOD=0x01;
	TH0=(65536-9174)/256;   //10ms
	TL0=(65536-9174)%256;
	EA=1;
	ET0=1;
	TR0=1;
}
void main(){
	Init();
	while(1){
		shi=num/10;
		ge=num%10;
		Showsmg(shi,ge);
	}
}
void Timer0() interrupt 1{
	TH0=(65536-9174)/256;   //重置
	TL0=(65536-9174)%256;
	t0++;
	if(t0==100){
		t0=0;
		num++;
		if(num==60)
			num=0;
	}
}
  • 独立按键的检测
//按键检测
void KeyDown(){
	GPIO_KEY=0x0f;     //独立按键端口
	if(GPIO_KEY!=0x0f){
		Delay(10);
		if(GPIO_KEY!=0x0f){
			GPIO_KEY=0x0f;
			switch(GPIO_KEY){
				case 0x0d : Key0();break;         //k0
				case 0x0e : Key1(flag);break;         //k1
				case 0x0b : Key2(flag);break;         //k2
				case 0x07 : Key3();break;         //k3
			}
		}while(GPIO_KEY!=0x0f);  //松手检测
	}
}
  • 独立按键处理函数
void Key0(){
	TR0=0;
	if(flag==0)   //第一次按K0
		flag=1;
	else          //第二次按K0
		flag=2;
}
void Key1(u8 flag){
	switch(flag){
		case 0: break;
		case 1: num+=1;if(num>=60) num=0;break;     //num超过59处理
		case 2: num+=10;if(num>60) num-=60;break;   //十位大于5处理
	}
}
void Key2(u8 flag){
	switch(flag){
		case 0: break;
		case 1: num-=1;if(num>59) num=59;break;
		case 2: {if(num>9) num-=10;
				else num+=50;}break;
	}
}
void Key3(){
	TR0=1;    //定时器启动
	flag=0;   //标志位清零
}

上面是每部分的代码,可以对应项目分析阅读,送上完整版。

smg.h

#include "reg52.h"
#include "intrins.h"

typedef unsigned int u16;
typedef unsigned char u8;
#define GPIO_KEY P3
u8 smgduan[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,
				   0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};

u8 t0,shi,ge,flag;
u16 num;				   
//38译码器					 
sbit LSA=P2^2;
sbit LSB=P2^3;
sbit LSC=P2^4;
//独立按键
sbit k0=P3^1;
sbit k1=P3^0;	
sbit k2=P3^2;	
sbit k3=P3^3;	
//数码管显示函数					 
void Showsmg(u8 decade,u8 unit);
//延时函数
void Delay(u16 i);
//初始化函数
void Init();
//按键检测
void KeyDown();				   
void Key0();
void Key1(u8 flag);
void Key2(u8 flag);
void Key3();

main.c

#include "smg.h"

void main(){
	Init();
	while(1){
		KeyDown();
		shi=num/10;
		ge=num%10;
		Showsmg(shi,ge);
	}
}

//数码管显示函数
void Showsmg(u8 shi,u8 ge){       //十位  个位
	//十位
	LSA=1;LSB=1;LSC=1;
	P0=smgduan[shi];
	Delay(1);
	P0=0x00;
	//个位
	LSA=0;LSB=1;LSC=1;
	P0=smgduan[ge];
	Delay(1);
	P0=0x00;
}
//延时函数
void Delay(u16 i)   //11.0592 1ms
{
	while(i--){
		unsigned char i, j;

		_nop_();
		i = 2;
		j = 199;
		do
		{
			while (--j);
		} while (--i);
	}
} 
//初始化函数
void Init(){
	t0=0;
	num=0;
	shi=0;
	ge=0;
	flag=0;
	TMOD=0x01;
	TH0=(65536-9174)/256;   //10ms
	TL0=(65536-9174)%256;
	EA=1;
	ET0=1;
	TR0=1;
}
//按键检测
void KeyDown(){
	GPIO_KEY=0x0f;
	if(GPIO_KEY!=0x0f){
		Delay(10);
		if(GPIO_KEY!=0x0f){
			GPIO_KEY=0x0f;
			switch(GPIO_KEY){
				case 0x0d : Key0();break;         //k0
				case 0x0e : Key1(flag);break;         //k1
				case 0x0b : Key2(flag);break;         //k2
				case 0x07 : Key3();break;         //k3
			}
		}while(GPIO_KEY!=0x0f);  //松手检测
	}
}

void Key0(){
	TR0=0;
	if(flag==0)   //第一次按K0
		flag=1;
	else          //第二次按K0
		flag=2;
}
void Key1(u8 flag){
	switch(flag){
		case 0: break;
		case 1: num+=1;if(num>=60) num=0;break;
		case 2: num+=10;if(num>60) num-=60;break;   //十位大于5处理
	}
}
void Key2(u8 flag){
	switch(flag){
		case 0: break;
		case 1: num-=1;if(num>59) num=59;break;
		case 2: {if(num>9) num-=10;
				else num+=50;}break;
	}
}
void Key3(){
	TR0=1;    //定时器启动
	flag=0;   //标志位清零
}
void Timer0() interrupt 1{
	TH0=(65536-9174)/256;   //重置
	TL0=(65536-9174)%256;
	t0++;
	if(t0==100){
		t0=0;
		num++;
		if(num==60)
			num=0;
	}
}

总结

首先先谢谢这位网友,这是间接的激励我呢。这是第一次在CSDN写博客让别人能找我帮忙,我刚到很开心哦,嘿嘿。也更加坚定我要好好学单片机的决心。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

51单片机应用篇-- --数码管60秒计时,独立按键可调 的相关文章

  • 【无标题】

    绘图控件GraphicsView 一 GraphicsView简介 1 QT有多种绘图相关的技术 xff0c 我们将在第2部分 2 4 QT绘图和图表 中比较详细系统的讲 2 本节简单讲一下GraphicsView的基本理论 xff0c 并
  • uboot源码分析之start.S解析

    1 start S引入 1 1 u boot lds中找到start S入口 1 在uboot中因为有汇编阶段参与 xff0c 因此不能直接找main c 整个程序的入口取决于链接脚本中ENTRY声明的地方 ENTRY start 因此 s
  • uboot启动第二阶段

    uboot启动第二阶段 start armboot函数简介 一个很长的函数 1 这个函数在uboot lib arm board c的第444行开始到908行结束 2 450行还不是全部 xff0c 因为里面还调用了别的函数 3 为什么这么
  • cmake设置编译类型为release命令

    cmake编译类型通常默认为debug xff0c 但是在编译软件时 xff0c 一般都需要使用release版本的 xff0c debug太慢了 设置为release版本可以在cmake文件里进行 xff0c 也可以在运行cmake命令时
  • 设计模式之单例模式(Singleton),以C++为例,实现日志输出。

    Hello大家好 xff0c 好久没更新了 xff0c 今天给大家补上最基础的设计模式 xff1a 单例模式 这个单例模式实在是我的心结啊 xff0c 2021年末左右面试京东算法岗 xff0c 面试官让我写一个单例 xff0c 没写出来
  • 源码分析MyBatis对数值(int、double)类型进行test判断的误区

    文章目录 问题描述问题分析验证解析表达式执行解析后表达式分别测试两个条件 查询Ognl官方文档验证问题解决 问题描述 在如下判断中 xff0c 如果type类型为int xff0c 那么对于type 61 39 39 部分判断会出现一些问题
  • Git报错:error: xxxx bytes of body are still expected.

    git一个很老的项目 xff0c 项目深度很深 xff0c 报错 xff1a error 7857 bytes of body are still expected fetch pack unexpected disconnect whil
  • 设计模式之代理模式(Proxy),以C++为例,实现远程代理、虚拟代理、保护代理等。

    兄弟姐妹们好 xff0c 又是好久没有更新了 xff0c 今天给大家简单介绍代理模式 xff0c 一个很简单的设计模式 xff0c 旨在不改变原对象的情况下通过代理对象来控制对原对象的访问 代理模式根据具体情况还可以分为远程代理 虚拟代理
  • C++ 互斥锁原理以及实际使用介绍

    兄弟姐妹们 xff0c 我又回来了 xff0c 今天带来实际开发中都需要使用的互斥锁的内容 xff0c 主要聊一聊如何使用互斥锁以及都有哪几种方式实现互斥锁 实现互斥 xff0c 可以有以下几种方式 xff1a 互斥量 xff08 Mute
  • 【C++】使用【windwos api】获取windwos计算机的基本信息

    今天来一篇获取windows计算机的基本信息的文章 xff0c 包含计算机名称 操作系统版本 处理器信息 内存信息 硬盘信息 显示器信息 网络信息 驱动程序信息 电源信息 其他硬件信息 目录 一 windwos系统包含的基本信息 二 获取信
  • C++ POCO库的基础介绍(Windwos和Linux)

    简单介绍C 43 43 POCO库能干什么 xff0c 后续有时间的话将根据其每个点详细解析 xff0c 关注我 本篇包含POCO库简单介绍 下载以及安装方式 简单代码示例 目录 一 POCO简单介绍 1 1 POCO库的基本模块 1 2
  • ROS踩坑记录

    ROS踩坑记录 问题 xff1a ubuntu 没有 dev ttyUSB0问题 xff1a 运行 launch 文件或 ROS 节点时出现 exit code 9 错误提示问题 xff1a windows使用vscode远程连接 xff0
  • STM32串口数据接收 --环形缓冲区

    STM32串口数据接收 环形缓冲区 环形缓冲区简介 在单片机中串口通信是我们使用最频繁的 xff0c 使用串口通信就会用到串口的数据接收与发送 xff0c 环形缓冲区方式接收数据可以更好的保证数据丢帧率第 在通信程序中 xff0c 经常使用
  • 如何设计安全可靠的开放接口---对请求参加密保护

    文章目录 如何设计安全可靠的开放接口 系列前言AES加解密代码实现 如何设计安全可靠的开放接口 系列 1 如何设计安全可靠的开放接口 之Token 2 如何设计安全可靠的开放接口 之AppId AppSecret 3 如何设计安全可靠的开放
  • rosdep init报错解决方法

    rosdep init报错解决方法 很多小伙伴在安装ROS的过程中都不可避免的要执行rosdep init和rosdep update这两行命令行 xff0c 这也是在安装ROS的过程中最让人头疼的两步 xff0c 一般都没法一次成功 xf
  • NVIDIA Jetson Nano/Xavier NX 扩容教程

    在售的 NVIDIA Jetson 内置 16 GB 的 eMMC xff0c 并已安装了 ubuntu 18 04 LTS 和 NVIDIA JetPack 4 6 xff0c 所以剩余的用户可用空间大约 2GB xff0c 这对将 NV
  • 深度学习框架YOLOv3的C++调用

    深度学习框架YOLOv3的C 43 43 调用 深度学习框架YOLOv3的C 43 43 调用 xff08 1 xff09 tensorflow版本的YOLOv3的C 43 43 调用 xff08 失败 xff09 xff08 2 xff0
  • 基于GPT-2实现图像文本生成

    原理 使用GPT 2模型处理文本 xff0c 做decoder 使用google的vit base patch16 224模型处理图像 xff0c 做encoder 最后通过VisionEncoderDecoderModel将这两个模型粘起
  • C语言中常见的两个比较字符串是否相等的函数strcmp和strncmp

    函数 xff1a strcmp和strncmp strcmp 使用格式 xff1a include lt string h gt int strcmp const char s1 const char s2 设这两个字符串为str1 xff
  • sprintf和printf 用法的区别

    printf 的作用是标准化输出 xff0c 默认的对象是标准输出缓冲区 xff0c 要有一定的条件才能把缓冲区里面的数据输出 sprintf 作用是格式化输出函数 xff0c 保存字符串到缓冲区中 xff0c 起到拼接字符串的作用 功能

随机推荐

  • 第六篇,STM32脉冲宽度调制(PWM)编程

    1 PWM概念 PWM叫脉冲宽度调制 Pulse Width Modulation xff0c 通过编程控制输出方波的频率和占空比 高低电平的比例 xff0c 广泛应用在测量 xff0c 通信 xff0c 功率控制等领域 呼吸灯 xff0c
  • 第十篇,STM32串口蓝牙编程

    1 串口蓝牙概念 串口蓝牙是一个蓝牙模块 xff0c 内部有蓝牙模块和程序 xff0c 可以进行蓝牙通信 xff0c 同时提供一个串口接口 xff0c 通过串口可以配置蓝牙模块进行数据传输 2 使用串口3连接蓝牙模块 3 手机上安装蓝牙调试
  • LeetCode岛屿问题通用解决模板

    文章目录 前言第一题 xff1a 求岛屿的周长模板整理遍历方向确定边界重复遍历问题处理 模板解第一题第二题 xff1a 求岛屿数量第三题 xff1a 岛屿的最大面积第四题 xff1a 统计子岛屿第五题 xff1a 统计封闭岛屿的数目第六题
  • 第十四篇,STM32的CAN总线通信

    1 CAN总线的概念 CAN指的是控制器局域网网络 Controller Area Network xff0c 由德国博世汽车电子厂商开发出来 CAN使用差分信号 xff0c 具有较强的抗干扰能力和传输稳定性 CAN属于多主通信 xff0c
  • OpenCV图像处理学习十九,像素重映射cv::remap

    一 像素重映射概念 重映射就是把输入图像中各个像素按照制定的规则顺序映射到另外一张图像的对应位置上去 xff0c 形成一张新的图像 二 像素映射API函数接口 cv remap xff08 InputArray src 输入图像 Outpu
  • OpenCV图像处理学习二十一,直方图比较方法

    一 直方图比较 直方图比较是对输入的两张图像进行计算得到直方图H1与H2 xff0c 归一化到相同的尺度空间 xff0c 然后可以通过计算H1与H2的之间的距离得到两个直方图的相似程度 xff08 每张图像都有唯一的直方图与之对应 xff0
  • 嵌入式FreeRTOS学习九,任务链表的构成,TICK时间中断和任务状态切换调度

    一 tskTaskControlBlock 函数结构体 在tskTaskControlBlock 任务控制块结构体中 xff0c 其中有任务状态链表和事件链表两个链表成员 xff0c 首先介绍任务状态链表这个结构 xff0c 这个链表通常用
  • SOAP传输协议

    一 HTTP传输协议 超文本传输协议 xff08 HyperText Transfer Protocol xff0c 缩写 xff1a HTTP xff09 xff0c 它是基于请求 响应的模式协议 xff0c 客户端发出请求 xff0c
  • ONVIF简介

    一 什么是ONVIF ONVIF规范描述了网络视频的模型 接口 数据类型以及数据交互的模式 并复用了一些现有的标准 xff0c 如WS系列标准等 ONVIF规范的目标是实现一个网络视频框架协议 xff0c 使不同厂商所生产的网络视频产品 x
  • gsoap工具生成onvif设备搜索(remotediscovery)代码框架

    什么是gsoap工具 xff1f gSOAP 提供了两个工具来方便开发人员使用 C C 43 43 语言快速开发Web 服务应用 xff0c 通过 gSOAP 提供的这两个工具 xff0c 开发人员可以快速生成服务端与客户端代码框架 xff
  • Latex之给字符上加横线、波浪等

    Latex 前几天想在 x x x 上加波浪号 xff0c 一时间忘记怎么打 xff0c 现在记录下来 xff0c 以后好查阅 加 号 xff1a hat x 加横线 xff1a overline x 加宽 xff1a widehat x
  • 数据结构笔记-2(线性表)

    线性表 2 1 线性表 1 定义 是零个或多个具有相同类型的数据元素的有序数列 xff1b xff08 长度等于零的线性表为空表 xff09 非空线性表通常记为 xff1a L xff1d a 1 xff0c a 2 xff0c xff0c
  • 数据结构-6(图)

    图 图的逻辑结构 图的定义 xff1a 图是由顶点的有穷非空集合和顶点之间边的集合组成 xff0c 通常表示为 xff1a G 61 V xff0c E 其中 xff1a G表示一个图 xff0c V是图G中顶点的集合 xff0c E是图G
  • 【leetcode常见面试题】螺旋矩阵解题思路

    文章目录 螺旋矩阵解题思路先找行进路线找每条路线的结束位置再找每条路线的结束位置模拟行走 螺旋矩阵 II总结 螺旋矩阵 解题思路 本题可以采用模拟的方式 xff0c 设4种行走方向 xff0c 如下图 xff1a 先找行进路线 4个方向的行
  • C++面向对象程序设计学习心得

    C 43 43 面向对象程序设计学习心得 经过几周c 43 43 面向对象程序设计的学习 xff0c 对面向对象程序设计有了一些了解 递归 简单地讲 xff0c 递归就是程序直接或间接调用本身的编程技巧 xff0c 通过把一个不能或不好解决
  • STL学习心得

    STL概述 STL组件 1 容器 xff08 Container xff09 xff0d 管理某类对象的集合 2 迭代器 xff08 Iterator xff09 xff0d 在对象集合上进行遍历 xff08 注意 xff1a 这些集合可能
  • 安装nodejs和vue出现问题

    安装nodejs国内镜像时报错 npm install g cnpm registry 61 https registry npm taobao org不知道这是啥错误 xff0c 怎么改啊 xff1f 安装vue 从官网点击下载 下载后点
  • Jetson nano/nx通过网线连接电脑实现远程控制

    Jetson nano nx通过网线连接电脑实现远程控制 摘要1 nano nx桌面共享设置2 安装dconf editor解除加密3 自启VNC server4 网络共享5 获取IP地址6 安装PuTTy7 安装VNC Viewwer8
  • 一文解决MySQL突击面试,关键知识点总结

    文章目录 MySQL重要知识点回顾一 索引1 为什么需要索引2 索引的结构3 避免索引失效3 1 联合索引不满足最左匹配原则3 2 隐式转换3 3 like查询3 4 索引列存在运算或者使用函数3 5 优化器 4 执行计划4 1 type4
  • 51单片机应用篇-- --数码管60秒计时,独立按键可调

    开篇先说一句废话 本旺名字叫萨摩耶 xff0c xff0c Please 叫我旺财 xff0c xff0c xff0c 哈哈 xff0c 招财进宝嘛 xff01 缘由 本来按照我的学习计划 xff0c 我现在应该是单片机的学习过程 xff0