51单片机入门——(新)简易数字时钟

这篇具有很好参考价值的文章主要介绍了51单片机入门——(新)简易数字时钟。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

设计要求

  1. 实现正确稳定地显示小时(两位数)、分钟(两位数)、秒钟(两位数),同时数码管应无闪烁问题。
  2. 通过按键分别实现时、分信息的调整,方便用户对时间的校准。
  3. 加入闹铃功能在(本设计中用LED代替)。

原理图

51单片机入门——(新)简易数字时钟

按键部分介绍

51单片机入门——(新)简易数字时钟
key1用于切换时分秒的加减。例:第一次按下后,按key2\key3时“秒”加\减,第二次按下后,按key2\key3时“分”加\减,第三次按下后,按key2\key3时“时”加\减,第四次按下后,按key2\key3不起作用,依次循环。

key2用于加。

key3用于减。

key4用于切换时间显示和闹钟显示切换。

代码解析

mian.c

#include <reg52.h>
#include "SMG.H"

sbit KEY1 = P3^0;
sbit KEY2 = P3^1;
sbit KEY3 = P3^2;
sbit KEY4 = P3^3;  
sbit LED = P3^4;

uchar T0RH = 0 ; // T0 重载值的高字节
uchar T0RL = 0 ; // T0 重载值的低字节

uchar sec =10, min = 10, hour = 10; // 初始化时分秒
uchar alarmSec, alarmMin, alarmHour; // 闹铃时间
bit alarm = 0; // 闹铃标志位
bit key1,key2,key3,key4; // 按键标志位
uchar b = 0; // 切换标志位

void ConFigTimer0(uchar ms);
void LedDriver();
void KeyDriver();

void main()
{
	EA = 1;
	ConFigTimer0(2);
	while (1)
	{
		LedDriver();
		KeyDriver();
		(alarmMin == min && alarmHour == hour) ? (LED = 1) : (LED = 0);
	}
}

void LedDriver()
{
	static unsigned char i;
	if (!alarm)
	{
		switch (i)
		{
			case 0 : ledBuff[0] = ledChar[sec % 10]; break;
			case 1 : ledBuff[1] = ledChar[sec / 10]; break;
			case 2 : ledBuff[2] = ledChar[10]; break;
			case 3 : ledBuff[3] = ledChar[min % 10]; break;
			case 4 : ledBuff[4] = ledChar[min / 10]; break;
			case 5 : ledBuff[5] = ledChar[10]; break;
			case 6 : ledBuff[6] = ledChar[hour % 10]; break;
			case 7 : ledBuff[7] = ledChar[hour / 10]; break;			
		}	
	}
	else
	{
		switch (i)
		{
			case 0 : ledBuff[0] = ledChar[alarmSec % 10]; break;
			case 1 : ledBuff[1] = ledChar[alarmSec / 10]; break;
			case 2 : ledBuff[2] = ledChar[10]; break;
			case 3 : ledBuff[3] = ledChar[alarmMin % 10]; break;
			case 4 : ledBuff[4] = ledChar[alarmMin / 10]; break;
			case 5 : ledBuff[5] = ledChar[10]; break;
			case 6 : ledBuff[6] = ledChar[alarmHour % 10]; break;
			case 7 : ledBuff[7] = ledChar[alarmHour / 10]; break;			
		}
	}
	i ++;
	i &= 0x07;
}
	
void KeyDriver()
{

	if (KEY1 == 0 && key1 == 0)
	{
		key1 = 1;
	}
	else if (KEY1 == 1 && key1 == 1)
	{
		key1 = 0;
		b ++;
		b &= 0x03;
	}

	if (KEY2 == 0 && key2 == 0)
	{
		key2 = 1;
	}
	else if (KEY2 == 1 && key2 == 1 )
	{
		key2 = 0;
		if(!alarm)
		{
			switch (b)
			{
				case 1 : sec ++; break;
				case 2 : min ++; break;
				case 3 : hour ++; break;
			}
		}
		else
		{
			switch (b)
			{
				case 1 : alarmSec ++; break;
				case 2 : alarmMin ++; break;
				case 3 : alarmHour ++; break;
			}
		}	
	}

	if (KEY3 == 0 && key3 == 0)
	{
		key3 = 1;
	}
	else if (KEY3 == 1 && key3 == 1 )
	{
		key3 = 0;
		if(!alarm)
		{
			switch (b)
			{
				case 1 : sec --; break;
				case 2 : min --; break;
				case 3 : hour --; break;
			}
		}
		else
		{
			switch (b)
			{
				case 1 : alarmSec --; break;
				case 2 : alarmMin --; break;
				case 3 : alarmHour --; break;
			}
		}	
	}

	if (KEY4 == 0 && key4 == 0)
	{
		key4 = 1;
	}
	else if (KEY4 == 1 && key4 == 1)
	{
		key4 = 0;
		alarm = ~alarm;
	}

}

/* 配置并启动T0 ,12MHz */
void ConFigTimer0(uchar ms)
{
	ulong tmp ;

	tmp = 12000000 / 12;
	tmp = (tmp * ms) / 1000;
	tmp = 65536 - tmp;
	tmp += 2;
	T0RH = (uchar)(tmp >> 8);
	T0RL = (uchar)tmp;
	TMOD &= 0xF0;
	TMOD |= 0x01;
	TH0 = T0RH;
	TL0 = T0RL;
	ET0 = 1;
	TR0 = 1;
}

void InterruptTimer0() interrupt 1 
{
	uint i;
	TH0 = T0RH;
	TL0 = T0RL;
	LedScan();
	i ++;
	if (i >= 500)
	{
		i = 0;
		sec ++;
		if (sec >= 60)
		{
			sec = 0;
			min ++;
			if (min >= 60)
			{
				min = 0;
				hour ++;
				if (hour >= 24)
					hour = 0;
			}
		}
	}
} 

SMG.H

#ifndef _SMG_H_
#define _SMG_H_

#include<REG52.h>

typedef unsigned char uchar ;
typedef unsigned int uint ;
typedef	unsigned long ulong ;

extern uchar ledChar[12], ledBuff[8];

extern void LedScan();

#endif

SMG.C

#include"SMG.H"

uchar ledChar[12] = { //共阴极数码管显示字符转换表
	0x3F, 0x06, 0x5B, 0x4F, 0x66, 0x6D, 0x7D, 0x07, 0x7F, 0x6F, 0x40, 0x00
};

uchar ledBuff[8] = { //数码管显示缓冲区
 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00
};

/* 数码管动态扫描刷新函数,需在定时中断中调用 */
void LedScan()
{
	static uchar i = 0; //动态扫描的索引
 想·
	P0 = ledChar[11]; //显示消隐
	P2 = ~(0x80 >> i);
	P0 = ledBuff[i];
	i ++;
	i &= 0x07;
}

在学习数码管动态扫描的时候,为了方便大家理解,我们程序写的细致一些,给大家引入了 switch 的用法,随着编程能力与领悟能力的增强,我们在编程上也可以改进一下逻辑算法,让程序变的更简洁。这种逻辑算法,通常不是靠学一下可以全部掌握的,而是通过不断的编写程序以及研究他人程序的过程中一点点积累起来的。

前边动态扫描刷新函数我们是这么写的:

P0 = 0x00;
switch (i)
	{
		case 0 : P2 = 0xfe; P0 = ledTemp[0]; break;
		case 1 : P2 = 0xfd; P0 = ledTemp[1]; break;
		case 2 : P2 = 0xfb; P0 = ledTemp[2]; break;
		case 3 : P2 = 0xf7; P0 = ledTemp[3]; break;
		case 4 : P2 = 0xef; P0 = ledTemp[4]; break;
		case 5 : P2 = 0xdf; P0 = ledTemp[5]; break;
		case 6 : P2 = 0xbf; P0 = ledTemp[6]; break;
		case 7 : P2 = 0x7f; P0 = ledTemp[7]; break;
	}
	i ++;
	i &= 0x07;

我们来分析每一个 case 分支,它们的结构是相同的,即改变 P2、改变索引 i、取数据写入 P0,只要把 case 后的常量与 P2和 LedBuff 的下标对比,就可以发现它们其实是相等的,那么我们可以直接把常量值(实际上就是 i 在改变前的值)赋值给它们即可,而不必写上 6 遍。还剩下一个 i 的操作,它进行了 7 次相同的++与一次归 0 操作,那么很明显用++和 if 判断就可以替代这些操作。下面就是我们据此改进后的代码:

	P0 = 0x00; //显示消隐
	P2 = ~(0x80 >> i);
	P0 = ledBuff[i];
	i ++;
	i &= 0x07;

大家看一下,P2 = ~(0x80 >> i);这行代码就利用了上面讲到的~和>>,利用右移运算符来实现1的右移,因为我们使用的是共阴数码管,所以有一个取反来实现0的右移,而 P0 的赋值也只需要一行代码,i 的处理也很简单。这样写成的代码是不是要简洁的多,也巧妙的多,而功能与前面的 switch 是一样的,同样可以完美实现动态显示刷新的功能。

而且一般的if在该代码中我们也换成了&=,例如i &= 0x07;b &= 0x03;在这里我是要让b 在 0~3 之间变化,加到 4 就自动归零,i在0~7之间变化,加到8就自动清零按照常规你可以用前面讲过的 if 语句轻松实现,但是你现在看一下这样程序是不是同样可以做到这一点呢?因为 0、1、2、3 这四个数值正好占用 2 个二进制的位(0~7刚好占3个),所以我们把一个字节的高 6 (5)位一直清零的话,这个字节的值自然就是一种到 4 (8)归零的效果了。看一下,这样一句代码比 if 语句要更为简洁吧,而效果完全一样。文章来源地址https://www.toymoban.com/news/detail-508884.html

到了这里,关于51单片机入门——(新)简易数字时钟的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 51单片机数字时钟

    我们学单片机,一般都会写数字时钟编程。它能帮助我们: a. 能够系统性地总结掌握的知识,将单元模块知识有机的结合在一起。 b. 能够充分协调好硬件与软件之间的相互结合,合理设计硬件电路。 c. 掌握单片机 C 语言判断语句、分支语句以及子程序调用等编程知识。

    2024年02月11日
    浏览(41)
  • 使用51单片机和DS1302时钟芯片做一个简易的电子时钟

    今天给大家推荐一个51单片机小实验,带你使用51单片机做一款简易的电子时钟,其中计时模块采用DS1302硬件模块,显示采用LCD显示屏,具体怎么实现开来一起看看吧! DS1302 是 DALLAS 公司推出的涓流充电时钟芯片,内含有一个实时时钟和31字节静态 RAM,通过简单的串行接口与

    2024年02月11日
    浏览(46)
  • 51单片机制作简易数字电压表

    首先打开proteus软件导入各个元器件,并连线。  接下来介绍下adc0808的导入方法  频率发生器的导入方法:  接下来就要编写c程序了,代码都是经过测试的,可以方型复制使用。 最后就是编译成hex文件并导入proteus中仿真了,然后调节滑动变阻器,就可以发现电压表的值和l

    2024年02月11日
    浏览(39)
  • 51单片机的数字时钟系统【含仿真+程序+报告+原理图】

    该系统由AT89C51单片机+DS1302时钟模块+按键模块+LCD显示模块构成。 利用51单片机实现电子时钟的功能。使用DS1302芯片作为计时设备,用LCD1602作为显示设备,按键按下可修改当前时间,按下的时候蜂鸣器会di一声。 可实现基本功能: 1、实时显示当前时间和日期 按键功能: 1、点

    2024年02月06日
    浏览(51)
  • 51单片机简易时钟闹钟八位数码管显示仿真( proteus仿真+程序+原理图+报告+讲解视频)

    51单片机简易时钟闹钟八位数码管显示仿真( proteus仿真+程序+原理图+报告+讲解视频) 仿真图proteus7.8及以上 程序编译器:keil 4/keil 5 编程语言:C语言 设计编号:S0046 本设计旨在设计一个基于51单片机的多功能数字时钟闹钟proteus仿真设计,可以显示时、分、秒,并且可以设置时

    2024年02月09日
    浏览(42)
  • 单片机原理与应用课程设计-基于51单片机的时钟日历

    摘 要 本课程设计是基于51单片机的日历时钟设计。作为嵌入式系统中常用的控制器,单片机在各种电子设备和系统中广泛应用。日历时钟作为一个常见的功能模块,在现代生活中具有重要意义。因此,设计一个基于51单片机的日历时钟,不仅有助于我们掌握单片机编程技术和

    2024年02月20日
    浏览(66)
  • 基于51单片机的简易电梯系统的设计

      系统概述 设计要求 根据所学的知识以及技能,利用MCS-51系列单片机为中心设计一个简易电梯系统,实现四层电梯的无故障运行,并用数码管和指示灯显示对应楼层和运行方向等基本信息。设计所用的单片机为STC89C52为主要的控制器,自带AD转换,当一边有按键按下时,另一

    2024年02月09日
    浏览(50)
  • 51单片机简易电阻测量仪仿真设计

    51单片机简易电阻测量仪仿真设计( proteus仿真+程序+报告+讲解视频) 仿真图proteus7.8及以上 程序编译器:keil 4/keil 5 编程语言:C语言 设计编号:S0037 基于51单片机的简易电阻测量仪仿真设计( proteus仿真+程序+报告+讲解视频) 单片机最小系统,或者称为 最小应用系统,是指用最

    2023年04月26日
    浏览(45)
  • 基于51单片机、DS1302时钟模块的电子闹钟设计

    提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 提示:以下是本篇文章正文内容,下面案例可供参考 DS1302 是美国DALLAS公司推出的一种高性能、低功耗、带RAM的实时时钟电路,它可以对年、月、日、周、时、分、秒进行计时,具有闰年补偿功能,工作

    2024年02月02日
    浏览(63)
  • 基于AT89C51单片机的电子时钟设计

    点击链接获取Keil源码与Project Backups仿真图: https://download.csdn.net/download/qq_64505944/87695258?spm=1001.2014.3001.5503 源码获取 主要内容: 1.设计出电子数字钟的电路,并用protus进行仿真画出对应的电路图 2.设计出电子数字钟的源程序,并用Keil进行编辑生成HEX文件 3.在protus中进行测试。

    2024年02月09日
    浏览(63)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包