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

开篇先说一句废话····
本旺名字叫萨摩耶,,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写博客让别人能找我帮忙,我刚到很开心哦,嘿嘿。也更加坚定我要好好学单片机的决心。

  • 31
    点赞
  • 155
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
下面是一个基于C51单片机独立按键控制数码管的例子。 ``` #include <reg51.h> //包含头文件 #define uchar unsigned char //宏定义uchar为unsigned char sbit LSA=P2^2; //定义数码管连接的引脚,从左到右分别为LSA、LSB、LSC、DIO、CLK sbit LSB=P2^3; sbit LSC=P2^4; sbit DIO=P2^0; sbit CLK=P2^1; uchar code table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f}; //定义显示0~9的编码 void delay(uchar i) //延时函数 { uchar j,k; for(j=i;j>0;j--) { for(k=110;k>0;k--); } } void main() { uchar i=0,j=0,k=0; //定义三个计数器 while(1) //循环 { if(P3!=0xff) //判断是否有按键按下 { delay(5); //消除抖动 if(P3!=0xff) //再次判断是否有按键按下 { switch(P3) //根据按键进行操作 { case(0xfe): //第一个按键按下 i++; //计数器i加1 if(i>9) //当计数器i大于9时,i清零 { i=0; } break; case(0xfd): //第二个按键按下 j++; //计数器j加1 if(j>9) //当计数器j大于9时,j清零 { j=0; } break; case(0xfb): //第三个按键按下 k++; //计数器k加1 if(k>9) //当计数器k大于9时,k清零 { k=0; } break; default: break; } } } LSA=0; //锁存器清零 LSB=0; LSC=0; P0=table[i]; //P0口输出i对应的编码值 delay(5); //延时 LSA=1; //锁存器锁存 P0=0x00; //P0口清零 LSA=0; //锁存器清零 LSB=1; //同上 P0=table[j]; delay(5); LSB=0; LSC=1; //同上 P0=table[k]; delay(5); LSC=0; } } ``` 在该程序中,我们将P2口的第2、3、4、0、1个引脚分别连接到数码管的LSA、LSB、LSC、DIO、CLK引脚上。在程序中通过一个计数器来控制数码管的显示,同时通过三个独立按键来分别对三个计数器进行操作,从而实现对数码管控制

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值