最近在学单片机,就做一些小东西来玩玩,这次我做了一个闹钟,之前有做过一个红外遥控小车的,以后有时间的话上传一下。
目录
[最近在学单片机,就做一些小东西来玩玩,这次我做了一个闹钟,之前有做过一个红外遥控小车的,以后有时间的话上传一下。]
[序言:]
[用到的材料:]
[思路:]
[实物操作视频:]
[代码部分:]
[main.c]
[LCD1602.c LCD显示函数]
[LCD1602.h]
[DS1302.c 操控DS1302。]
[DS1302.h]
[Delay.c 延时函数]
[Delay.h]
[Timer0.c 定时器0。]
[Timer0.h]
[ShowTime.c 用来展示时钟的时间。]
[ShowTime.h]
[SetTime.c 用来设置实时时钟的时间。]
[SetTime.h]
[Buzzer.c 设置蜂鸣器响的模式]
[Buzzer.h]
[MatrixKey.c 矩阵按键]
[MatrixKey.h]
[SetAlarm.c 设立闹钟响铃时间]
[工程代码及示例视频网盘链接:]
序言:
最近在学习51单片机,刚好学过了DS1302实时时钟,就想着能不能结合之前学过的一些知识做一些实用的小东西,于是我就诞生了做一个闹钟的想法。
其实之前我学到定时器的时候就有这个想法了,不过经过的实验,定时器的精度太差,我就放弃了这个想法,DS1302的精度还是很高的,可以计时99年,记到天荒地老了。
用到的材料:
普中的STC89C52单片机,LCD1602。
思路:
用DS1302来显示实时时钟,将时间在LCD1602上显示出来,并通过按键和用户交互。设定能够调节实时时钟时间的函数,能够设定闹钟响铃时间的函数,当达到设定的响铃时间后,51单片机上的蜂鸣器发声30秒后关闭,模拟闹钟响铃。
实物操作视频:
【用51单片机制作一个闹钟】www.bilibili.com/video/BV1Fc…
代码部分:
main.c
#include <REGX52.H>
#include "LCD1602.h"
#include "DS1302.h"
#include "MatrixKey.h"
#include "Timer0.h"
#include "SetTime.h"
#include "ShowTime.h"
#include "Buzzer.h"
#include "SetAlarm.h"
unsigned char MODE;//选择的模式
void main()
{
LCD_Init();
DS1302_Init();
Timer0Init();
Buzzer_Init();//即初始化Timer1
LCD_ShowString(1,1," - - ");//静态字符初始化显示
LCD_ShowString(2,1," : : ");
DS1302_SetTime();//设置时间
while(1)
{
KeyNum=MatrixKey();//读取键码
if(KeyNum==1)
{
MODE++;
MODE%=3;
if(MODE==1||MODE==2)
{
SetTimeSelect=0;
SetAlarmSelect=0;
}
}
if(KeyNum==5)//保存调设的时间
{
DS1302_SetTime();
}
if(KeyNum==6)//退出到显示实时时间界面
{
MODE=0;
}
switch(MODE)//根据不同的功能执行不同的函数
{
case 0:
LCD_ShowString(1,11," ");
LCD_ShowString(2,11," ");
ShowTime();//显示实时时间
//到点就响铃
if(Alarm_Time[0]==DS1302_Time[0]
&&Alarm_Time[1]==DS1302_Time[1]
&&Alarm_Time[2]==DS1302_Time[2]
&&Alarm_Time[3]==DS1302_Time[3]
&&Alarm_Time[4]==DS1302_Time[4]
&&Alarm_Time[5]==DS1302_Time[5])//如果时间和设定的闹钟时间相同
{
AlarmWork();
}
break;
case 1:
LCD_ShowString(1,11,"Set");
LCD_ShowString(2,11,"Time");
SetTime();//调设实时时钟的时间
//到点就响铃
if(Alarm_Time[0]==DS1302_Time[0]
&&Alarm_Time[1]==DS1302_Time[1]
&&Alarm_Time[2]==DS1302_Time[2]
&&Alarm_Time[3]==DS1302_Time[3]
&&Alarm_Time[4]==DS1302_Time[4]
&&Alarm_Time[5]==DS1302_Time[5])//如果时间和设定的闹钟时间相同
{
AlarmWork();
}
break;
case 2:
LCD_ShowString(1,11,"Set");
LCD_ShowString(2,11,"Alarm");
SetAlarm();//调设闹钟的响铃时间
//到点就响铃
if(Alarm_Time[0]==DS1302_Time[0]
&&Alarm_Time[1]==DS1302_Time[1]
&&Alarm_Time[2]==DS1302_Time[2]
&&Alarm_Time[3]==DS1302_Time[3]
&&Alarm_Time[4]==DS1302_Time[4]
&&Alarm_Time[5]==DS1302_Time[5])
{
AlarmWork();
}
break;
}
}
}
void Timer0_Routine() interrupt 1
{
static unsigned int T0Count;
TL0 = 0x66; //设置定时初始值
TH0 = 0xFC; //设置定时初始值
T0Count++;
if(T0Count>=500)//每500ms进入一次
{
T0Count=0;
SetTimeFlashFlag=!SetTimeFlashFlag;//闪烁标志位取反
SetAlarmFlashFlag=!SetAlarmFlashFlag;//闪烁标志位取反
}
}
LCD1602.c LCD显示函数
LCD显示函数
#include <REGX52.H>
#include <INTRINS.h>
//引脚配置:
sbit LCD_RS=P2^6;
sbit LCD_RW=P2^5;
sbit LCD_EN=P2^7;
#define LCD_DataPort P0
//函数定义:
/**
* @brief LCD1602延时函数,11.0592MHz调用可延时1ms
* @param 无
* @retval 无
*/
void LCD_Delay()
{
unsigned char data i, j;
_nop_();
i = 2;
j = 199;
do
{
while (--j);
} while (--i);
}
/**
* @brief LCD1602写命令
* @param Command 要写入的命令
* @retval 无
*/
void LCD_WriteCommand(unsigned char Command)
{
LCD_RS=0;
LCD_RW=0;
LCD_DataPort=Command;
LCD_EN=1;
LCD_Delay();
LCD_EN=0;
LCD_Delay();
}
/**
* @brief LCD1602写数据
* @param Data 要写入的数据
* @retval 无
*/
void LCD_WriteData(unsigned char Data)
{
LCD_RS=1;
LCD_RW=0;
LCD_DataPort=Data;
LCD_EN=1;
LCD_Delay();
LCD_EN=0;
LCD_Delay();
}
/**
* @brief LCD1602设置光标位置
* @param Line 行位置,范围:1~2
* @param Column 列位置,范围:1~16
* @retval 无
*/
void LCD_SetCursor(unsigned char Line,unsigned char Column)
{
if(Line==1)
{
LCD_WriteCommand(0x80|(Column-1));
}
else if(Line==2)
{
LCD_WriteCommand(0x80|(Column-1+0x40));
}
}
/**
* @brief LCD1602初始化函数
* @param 无
* @retval 无
*/
void LCD_Init()
{
LCD_WriteCommand(0x38);//八位数据接口,两行显示,5*7点阵
LCD_WriteCommand(0x0c);//显示开,光标关,闪烁关
LCD_WriteCommand(0x06);//数据读写操作后,光标自动加一,画面不动
LCD_WriteCommand(0x01);//光标复位,清屏
}
/**
* @brief 在LCD1602指定位置上显示一个字符
* @param Line 行位置,范围:1~2
* @param Column 列位置,范围:1~16
* @param Char 要显示的字符
* @retval 无
*/
void LCD_ShowChar(unsigned char Line,unsigned char Column,char Char)
{
LCD_SetCursor(Line,Column);
LCD_WriteData(Char);
}
/**
* @brief 在LCD1602指定位置开始显示所给字符串
* @param Line 起始行位置,范围:1~2
* @param Column 起始列位置,范围:1~16
* @param String 要显示的字符串
* @retval 无
*/
void LCD_ShowString(unsigned char Line,unsigned char Column,char *String)
{
unsigned char i;
LCD_SetCursor(Line,Column);
for(i=0;String[i]!='\0';i++)
{
LCD_WriteData(String[i]);
}
}
/**
* @brief 返回值=X的Y次方
*/
int LCD_Pow(int X,int Y)
{
unsigned char i;
int Result=1;
for(i=0;i<Y;i++)
{
Result*=X;
}
return Result;
}
/**
* @brief 在LCD1602指定位置开始显示所给数字
* @param Line 起始行位置,范围:1~2
* @param Column 起始列位置,范围:1~16
* @param Number 要显示的数字,范围:0~65535
* @param Length 要显示数字的长度,范围:1~5
* @retval 无
*/
void LCD_ShowNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length)
{
unsigned char i;
LCD_SetCursor(Line,Column);
for(i=Length;i>0;i--)
{
LCD_WriteData(Number/LCD_Pow(10,i-1)%10+'0');
}
}
/**
* @brief 在LCD1602指定位置开始以有符号十进制显示所给数字
* @param Line 起始行位置,范围:1~2
* @param Column 起始列位置,范围:1~16
* @param Number 要显示的数字,范围:-32768~32767
* @param Length 要显示数字的长度,范围:1~5
* @retval 无
*/
void LCD_ShowSignedNum(unsigned char Line,unsigned char Column,int Number,unsigned char Length)
{
unsigned char i;
unsigned int Number1;
LCD_SetCursor(Line,Column);
if(Number>=0)
{
LCD_WriteData('+');
Number1=Number;
}
else
{
LCD_WriteData('-');
Number1=-Number;
}
for(i=Length;i>0;i--)
{
LCD_WriteData(Number1/LCD_Pow(10,i-1)%10+'0');
}
}
/**
* @brief 在LCD1602指定位置开始以十六进制显示所给数字
* @param Line 起始行位置,范围:1~2
* @param Column 起始列位置,范围:1~16
* @param Number 要显示的数字,范围:0~0xFFFF
* @param Length 要显示数字的长度,范围:1~4
* @retval 无
*/
void LCD_ShowHexNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length)
{
unsigned char i,SingleNumber;
LCD_SetCursor(Line,Column);
for(i=Length;i>0;i--)
{
SingleNumber=Number/LCD_Pow(16,i-1)%16;
if(SingleNumber<10)
{
LCD_WriteData(SingleNumber+'0');
}
else
{
LCD_WriteData(SingleNumber-10+'A');
}
}
}
/**
* @brief 在LCD1602指定位置开始以二进制显示所给数字
* @param Line 起始行位置,范围:1~2
* @param Column 起始列位置,范围:1~16
* @param Number 要显示的数字,范围:0~1111 1111 1111 1111
* @param Length 要显示数字的长度,范围:1~16
* @retval 无
*/
void LCD_ShowBinNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length)
{
unsigned char i;
LCD_SetCursor(Line,Column);
for(i=Length;i>0;i--)
{
LCD_WriteData(Number/LCD_Pow(2,i-1)%2+'0');
}
}
LCD1602.h
#ifndef __LCD1602_H__
#define __LCD1602_H__
//用户调用函数:
void LCD_Init();
void LCD_Delay();
void LCD_ShowChar(unsigned char Line,unsigned char Column,char Char);
void LCD_ShowString(unsigned char Line,unsigned char Column,char *String);
void LCD_ShowNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length);
void LCD_ShowSignedNum(unsigned char Line,unsigned char Column,int Number,unsigned char Length);
void LCD_ShowHexNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length);
void LCD_ShowBinNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length);
#endif
DS1302.c 操控DS1302。
操控DS1302。
#include <REGX52.H>
//引脚定义
sbit DS1302_SCLK=P3^6;
sbit DS1302_IO=P3^4;
sbit DS1302_CE=P3^5;
//寄存器写入地址/指令定义
#define DS1302_SECOND 0x80
#define DS1302_MINUTE 0x82
#define DS1302_HOUR 0x84
#define DS1302_DATE 0x86
#define DS1302_MONTH 0x88
#define DS1302_DAY 0x8A
#define DS1302_YEAR 0x8C
#define DS1302_WP 0x8E
//时间数组,索引0~6分别为年、月、日、时、分、秒、星期,设置为有符号的便于<0的判断
int DS1302_Time[]={23,10,25,20,56,15,3};
/**
* @brief DS1302初始化
* @param 无
* @retval 无
*/
void DS1302_Init(void)
{
DS1302_CE=0;
DS1302_SCLK=0;
}
/**
* @brief DS1302写一个字节
* @param Command 命令字/地址
* @param Data 要写入的数据
* @retval 无
*/
void DS1302_WriteByte(unsigned char Command,Data)
{
unsigned char i;
DS1302_CE=1;
for(i=0;i<8;i++)
{
DS1302_IO=Command&(0x01<<i);
DS1302_SCLK=1;
DS1302_SCLK=0;
}
for(i=0;i<8;i++)
{
DS1302_IO=Data&(0x01<<i);
DS1302_SCLK=1;
DS1302_SCLK=0;
}
DS1302_CE=0;
}
/**
* @brief DS1302读一个字节
* @param Command 命令字/地址
* @retval 读出的数据
*/
unsigned char DS1302_ReadByte(unsigned char Command)
{
unsigned char i,Data=0x00;
Command|=0x01; //将指令转换为读指令
DS1302_CE=1;
for(i=0;i<8;i++)
{
DS1302_IO=Command&(0x01<<i);
DS1302_SCLK=0;
DS1302_SCLK=1;
}
for(i=0;i<8;i++)
{
DS1302_SCLK=1;
DS1302_SCLK=0;
if(DS1302_IO){Data|=(0x01<<i);}
}
DS1302_CE=0;
DS1302_IO=0; //读取后将IO设置为0,否则读出的数据会出错
return Data;
}
/**
* @brief DS1302设置时间,调用之后,DS1302_Time数组的数字会被设置到DS1302中
* @param 无
* @retval 无
*/
void DS1302_SetTime(void)
{
DS1302_WriteByte(DS1302_WP,0x00);
DS1302_WriteByte(DS1302_YEAR,DS1302_Time[0]/10*16+DS1302_Time[0]%10);//十进制转BCD码后写入
DS1302_WriteByte(DS1302_MONTH,DS1302_Time[1]/10*16+DS1302_Time[1]%10);
DS1302_WriteByte(DS1302_DATE,DS1302_Time[2]/10*16+DS1302_Time[2]%10);
DS1302_WriteByte(DS1302_HOUR,DS1302_Time[3]/10*16+DS1302_Time[3]%10);
DS1302_WriteByte(DS1302_MINUTE,DS1302_Time[4]/10*16+DS1302_Time[4]%10);
DS1302_WriteByte(DS1302_SECOND,DS1302_Time[5]/10*16+DS1302_Time[5]%10);
DS1302_WriteByte(DS1302_DAY,DS1302_Time[6]/10*16+DS1302_Time[6]%10);
DS1302_WriteByte(DS1302_WP,0x80);
}
/**
* @brief DS1302读取时间,调用之后,DS1302中的数据会被读取到DS1302_Time数组中
* @param 无
* @retval 无
*/
void DS1302_ReadTime(void)
{
unsigned char Temp;
Temp=DS1302_ReadByte(DS1302_YEAR);
DS1302_Time[0]=Temp/16*10+Temp%16;//BCD码转十进制后读取
Temp=DS1302_ReadByte(DS1302_MONTH);
DS1302_Time[1]=Temp/16*10+Temp%16;
Temp=DS1302_ReadByte(DS1302_DATE);
DS1302_Time[2]=Temp/16*10+Temp%16;
Temp=DS1302_ReadByte(DS1302_HOUR);
DS1302_Time[3]=Temp/16*10+Temp%16;
Temp=DS1302_ReadByte(DS1302_MINUTE);
DS1302_Time[4]=Temp/16*10+Temp%16;
Temp=DS1302_ReadByte(DS1302_SECOND);
DS1302_Time[5]=Temp/16*10+Temp%16;
Temp=DS1302_ReadByte(DS1302_DAY);
DS1302_Time[6]=Temp/16*10+Temp%16;
}
DS1302.h
#ifndef __DS1302_H__
#define __DS1302_H__
//外部可调用时间数组,索引0~6分别为年、月、日、时、分、秒、星期,设置为有符号的便于<0的判断
extern int DS1302_Time[];
void DS1302_Init(void);
void DS1302_WriteByte(unsigned char Command,Data);
unsigned char DS1302_ReadByte(unsigned char Command);
void DS1302_SetTime(void);
void DS1302_ReadTime(void);
#endif
Delay.c 延时函数
延时函数
/**
*@brief 延时函数,11.0592MHz
*@param xms,延时xms毫秒
*@retval 无
*/
void Delay(unsigned int xms)
{
unsigned char i, j;
while(xms--)
{
i = 2;
j = 239;
do
{
while (--j);
} while (--i);
}
}
Delay.h
#ifndef __DELAY_H__
#define __DELAY_H__
void Delay(unsigned int xms);
#endif
Timer0.c 定时器0。
定时器0。
#include <REGX52.H>
/**
* @brief 定时器0初始化,1毫秒@11.0592MHz
* @param 无
* @retval 无
*/
void Timer0Init(void)
{
TMOD &= 0xF0; //设置定时器模式
TMOD |= 0x01; //设置定时器模式
TL0 = 0x66; //设置定时初始值
TH0 = 0xFC; //设置定时初始值
TF0 = 0; //清除TF0标志
TR0 = 1; //定时器0开始计时
ET0=1;
EA=1;
PT0=0;
}
/*定时器中断函数模板
void Timer0_Routine() interrupt 1
{
static unsigned int T0Count;
TL0 = 0x66; //设置定时初始值
TH0 = 0xFC; //设置定时初始值
T0Count++;
if(T0Count>=1000)
{
T0Count=0;
}
}
*/
Timer0.h
#ifndef __TIMER0_H__
#define __TIMER0_H__
void Timer0Init(void);
#endif
ShowTime.c 用来展示时钟的时间。
用来展示时钟的时间。
#include <REGX52.H>
#include "DS1302.h"
#include "LCD1602.h"
/**
*@brief 显示时间函数
*@param 无
*@retval 无
*/
void ShowTime(void)//时间显示功能
{
DS1302_ReadTime();//读取时间
LCD_ShowNum(1,1,DS1302_Time[0],2);//显示年
LCD_ShowNum(1,4,DS1302_Time[1],2);//显示月
LCD_ShowNum(1,7,DS1302_Time[2],2);//显示日
LCD_ShowNum(2,1,DS1302_Time[3],2);//显示时
LCD_ShowNum(2,4,DS1302_Time[4],2);//显示分
LCD_ShowNum(2,7,DS1302_Time[5],2);//显示秒
}
ShowTime.h
#ifndef _SHOWTIME_H_
#define _SHOWTIME_H_
void ShowTime(void);//时间显示功能
#endif
SetTime.c 用来设置实时时钟的时间。
用来设置实时时钟的时间。
#include <REGX52.H>
#include "DS1302.h"
#include "LCD1602.h"
unsigned int KeyNum,SetTimeSelect,SetTimeFlashFlag;
/**
*@brief 时间设置函数
*@param 无
*@retval 无
*/
void SetTime(void)//时间设置功能
{
if(KeyNum==2)//按键2按下
{
SetTimeSelect++;//设置选择位加1
SetTimeSelect%=6;//越界清零
}
if(KeyNum==3)//按键3按下
{
DS1302_Time[SetTimeSelect]++;//时间设置位数值加1
if(DS1302_Time[0]>99){DS1302_Time[0]=0;}//年越界判断
if(DS1302_Time[1]>12){DS1302_Time[1]=1;}//月越界判断
if( DS1302_Time[1]==1 || DS1302_Time[1]==3 || DS1302_Time[1]==5 || DS1302_Time[1]==7 ||
DS1302_Time[1]==8 || DS1302_Time[1]==10 || DS1302_Time[1]==12)//日越界判断
{
if(DS1302_Time[2]>31){DS1302_Time[2]=1;}//大月
}
else if(DS1302_Time[1]==4 || DS1302_Time[1]==6 || DS1302_Time[1]==9 || DS1302_Time[1]==11)
{
if(DS1302_Time[2]>30){DS1302_Time[2]=1;}//小月
}
else if(DS1302_Time[1]==2)
{
if(DS1302_Time[0]%4==0)
{
if(DS1302_Time[2]>29){DS1302_Time[2]=1;}//闰年2月
}
else
{
if(DS1302_Time[2]>28){DS1302_Time[2]=1;}//平年2月
}
}
if(DS1302_Time[3]>23){DS1302_Time[3]=0;}//时越界判断
if(DS1302_Time[4]>59){DS1302_Time[4]=0;}//分越界判断
if(DS1302_Time[5]>59){DS1302_Time[5]=0;}//秒越界判断
}
if(KeyNum==4)//按键4按下
{
DS1302_Time[SetTimeSelect]--;//时间设置位数值减1
if(DS1302_Time[0]<0){DS1302_Time[0]=99;}//年越界判断
if(DS1302_Time[1]<1){DS1302_Time[1]=12;}//月越界判断
if( DS1302_Time[1]==1 || DS1302_Time[1]==3 || DS1302_Time[1]==5 || DS1302_Time[1]==7 ||
DS1302_Time[1]==8 || DS1302_Time[1]==10 || DS1302_Time[1]==12)//日越界判断
{
if(DS1302_Time[2]<1){DS1302_Time[2]=31;}//大月
if(DS1302_Time[2]>31){DS1302_Time[2]=1;}
}
else if(DS1302_Time[1]==4 || DS1302_Time[1]==6 || DS1302_Time[1]==9 || DS1302_Time[1]==11)
{
if(DS1302_Time[2]<1){DS1302_Time[2]=30;}//小月
if(DS1302_Time[2]>30){DS1302_Time[2]=1;}
}
else if(DS1302_Time[1]==2)
{
if(DS1302_Time[0]%4==0)
{
if(DS1302_Time[2]<1){DS1302_Time[2]=29;}//闰年2月
if(DS1302_Time[2]>29){DS1302_Time[2]=1;}
}
else
{
if(DS1302_Time[2]<1){DS1302_Time[2]=28;}//平年2月
if(DS1302_Time[2]>28){DS1302_Time[2]=1;}
}
}
if(DS1302_Time[3]<0){DS1302_Time[3]=23;}//时越界判断
if(DS1302_Time[4]<0){DS1302_Time[4]=59;}//分越界判断
if(DS1302_Time[5]<0){DS1302_Time[5]=59;}//秒越界判断
}
//更新显示,根据SetTimeSelect和SetTimeFlashFlag判断可完成闪烁功能
if(SetTimeSelect==0 && SetTimeFlashFlag==1){LCD_ShowString(1,1," ");}
else {LCD_ShowNum(1,1,DS1302_Time[0],2);}
if(SetTimeSelect==1 && SetTimeFlashFlag==1){LCD_ShowString(1,4," ");}
else {LCD_ShowNum(1,4,DS1302_Time[1],2);}
if(SetTimeSelect==2 && SetTimeFlashFlag==1){LCD_ShowString(1,7," ");}
else {LCD_ShowNum(1,7,DS1302_Time[2],2);}
if(SetTimeSelect==3 && SetTimeFlashFlag==1){LCD_ShowString(2,1," ");}
else {LCD_ShowNum(2,1,DS1302_Time[3],2);}
if(SetTimeSelect==4 && SetTimeFlashFlag==1){LCD_ShowString(2,4," ");}
else {LCD_ShowNum(2,4,DS1302_Time[4],2);}
if(SetTimeSelect==5 && SetTimeFlashFlag==1){LCD_ShowString(2,7," ");}
else {LCD_ShowNum(2,7,DS1302_Time[5],2);}
}
SetTime.h
#ifndef _SETTIME_H_
#define _SETTIME_H_
extern unsigned int KeyNum,SetTimeSelect,SetTimeFlashFlag;
void SetTime(void);//时间设置功能
#endif
Buzzer.c 设置蜂鸣器响的模式
设置蜂鸣器响的模式,由于51单片机上的蜂鸣器是无源蜂鸣器,控制起来有点麻烦,并且它的IO口还和LCD1602的IO口有部分冲突了,导致发出的声音有些奇怪,怎么说呢,就有点想蟋蟀的声音吧,本来我是想让它有像有规律的C4那样响的。
#include <REGX52.H>
#include "LCD1602.h"
#include "SetAlarm.h"
#include "DS1302.h"
sbit Buzzer=P2^5;
unsigned char Temp;
void Buzzer_Init(void) //100微秒@11.0592MHz蜂鸣器内部使用定时器
{
TMOD &= 0x0F; //设置定时器模式
TMOD |= 0x10; //设置定时器模式
TL1 = 0xA4; //设置定时初始值
TH1 = 0xFF; //设置定时初始值
TF1 = 0; //清除TF1标志
TR1 = 0; //定时器1停止计时
ET1=1;
EA=1;
PT1=0;
}
/**
*@brief 蜂鸣器内部使用定时器中断执行函数
*@param 无
*@retval 无
*/
void Timer1_Routine() interrupt 3
{
static unsigned int T1Count;
TL1 = 0xA4; //设置定时初始值
TH1 = 0xFF; //设置定时初始值
T1Count++;
if(T1Count<=1000)//一秒反转5次,即一秒钟响5次
{
Buzzer=~Buzzer;
}
else if(T1Count<=2000)
{
Buzzer=1;
}
if(T1Count==2000)
{
T1Count=0;
}
if(Alarm_Time[0]==DS1302_Time[0]
&&Alarm_Time[1]==DS1302_Time[1]
&&Alarm_Time[2]==DS1302_Time[2]
&&Alarm_Time[3]==DS1302_Time[3]
&&Temp==DS1302_Time[4]//用Temp来判断停止响铃时间(一分钟后)
&&Alarm_Time[5]==DS1302_Time[5])
{//一分钟后将定时器停了,蜂鸣器也就停了(定时器误差有点大,就不用定时器来定这个时间了)
TR1=0;
LCD_Init();//蜂鸣器和LCD1602共用了P2_5,冲突了,要初始化一下之后才能正常显示
LCD_Delay();
LCD_ShowString(1,1," - - ");//静态字符初始化显示
LCD_ShowString(2,1," : : ");
}
}
/**
*@brief 控制蜂鸣器的开始工作和停止工作
*@param 无
*@retval 无
*/
void AlarmWork()
{
TR1=1;
Temp=Alarm_Time[4]+1;//用Temp来判断闹钟停了的时间,这里设置一分钟后停止响铃
if(Temp==60)//越界判断
{
Temp=0;
}
}
Buzzer.h
#ifndef _BUZZER_H_
#define _BUZZER_H_
void Buzzer_Init(void);
void AlarmWork();
#endif
MatrixKey.c 矩阵按键
这部分是矩阵按键的代码,本来想用独立按键来设计的,但是发现按键数量有点少,实现对闹钟的控制有点复杂,于是我就放弃了用独立按键的方案,用按键数更多的矩阵按键。
#include <REGX52.H>
#include "Delay.h"
/**
* @brief 矩阵键盘读取按键键码
* @param 无
* @retval KeyNumber 按下按键的键码值
如果按键按下不放,程序会停留在此函数,松手的一瞬间,返回按键键码,没有按键按下时,返回0
*/
unsigned char MatrixKey()
{
unsigned char KeyNumber=0;
P1=0xFF;
P1_3=0;
if(P1_7==0){Delay(20);while(P1_7==0);Delay(20);KeyNumber=1;}
if(P1_6==0){Delay(20);while(P1_6==0);Delay(20);KeyNumber=5;}
if(P1_5==0){Delay(20);while(P1_5==0);Delay(20);KeyNumber=9;}
if(P1_4==0){Delay(20);while(P1_4==0);Delay(20);KeyNumber=13;}
P1=0xFF;
P1_2=0;
if(P1_7==0){Delay(20);while(P1_7==0);Delay(20);KeyNumber=2;}
if(P1_6==0){Delay(20);while(P1_6==0);Delay(20);KeyNumber=6;}
if(P1_5==0){Delay(20);while(P1_5==0);Delay(20);KeyNumber=10;}
if(P1_4==0){Delay(20);while(P1_4==0);Delay(20);KeyNumber=14;}
P1=0xFF;
P1_1=0;
if(P1_7==0){Delay(20);while(P1_7==0);Delay(20);KeyNumber=3;}
if(P1_6==0){Delay(20);while(P1_6==0);Delay(20);KeyNumber=7;}
if(P1_5==0){Delay(20);while(P1_5==0);Delay(20);KeyNumber=11;}
if(P1_4==0){Delay(20);while(P1_4==0);Delay(20);KeyNumber=15;}
P1=0xFF;
P1_0=0;
if(P1_7==0){Delay(20);while(P1_7==0);Delay(20);KeyNumber=4;}
if(P1_6==0){Delay(20);while(P1_6==0);Delay(20);KeyNumber=8;}
if(P1_5==0){Delay(20);while(P1_5==0);Delay(20);KeyNumber=12;}
if(P1_4==0){Delay(20);while(P1_4==0);Delay(20);KeyNumber=16;}
return KeyNumber;
}
MatrixKey.h
#ifndef __MATRIXKEY_H__
#define __MATRIXKEY_H__
unsigned char MatrixKey();
#endif
SetAlarm.c 设立闹钟响铃时间
这里是用来设立闹钟响铃时间的。
#include <REGX52.H>
#include "DS1302.h"
#include "LCD1602.h"
extern unsigned int KeyNum;
unsigned int SetAlarmSelect,SetAlarmFlashFlag;
int Alarm_Time[]={23,10,25,20,56,00};//对应年、月、日、时、分、秒,用于记录闹钟响的时间
void SetAlarm(void)//闹钟响铃时间设置功能
{
if(KeyNum==2)//按键2按下
{
SetAlarmSelect++;//设置选择位加1
SetAlarmSelect%=6;//越界清零
}
if(KeyNum==3)//按键3按下
{
Alarm_Time[SetAlarmSelect]++;//时间设置位数值加1
if(Alarm_Time[0]>99){Alarm_Time[0]=0;}//年越界判断
if(Alarm_Time[1]>12){Alarm_Time[1]=1;}//月越界判断
if( Alarm_Time[1]==1 || Alarm_Time[1]==3 || Alarm_Time[1]==5 || Alarm_Time[1]==7 ||
Alarm_Time[1]==8 || Alarm_Time[1]==10 || Alarm_Time[1]==12)//日越界判断
{
if(Alarm_Time[2]>31){Alarm_Time[2]=1;}//大月
}
else if(Alarm_Time[1]==4 || Alarm_Time[1]==6 || Alarm_Time[1]==9 || Alarm_Time[1]==11)
{
if(Alarm_Time[2]>30){Alarm_Time[2]=1;}//小月
}
else if(Alarm_Time[1]==2)
{
if(Alarm_Time[0]%4==0)
{
if(Alarm_Time[2]>29){Alarm_Time[2]=1;}//闰年2月
}
else
{
if(Alarm_Time[2]>28){Alarm_Time[2]=1;}//平年2月
}
}
if(Alarm_Time[3]>23){Alarm_Time[3]=0;}//时越界判断
if(Alarm_Time[4]>59){Alarm_Time[4]=0;}//分越界判断
if(Alarm_Time[5]>59){Alarm_Time[5]=0;}//秒越界判断
}
if(KeyNum==4)//按键4按下
{
Alarm_Time[SetAlarmSelect]--;//时间设置位数值减1
if(Alarm_Time[0]<0){Alarm_Time[0]=99;}//年越界判断
if(Alarm_Time[1]<1){Alarm_Time[1]=12;}//月越界判断
if( Alarm_Time[1]==1 || Alarm_Time[1]==3 || Alarm_Time[1]==5 || Alarm_Time[1]==7 ||
Alarm_Time[1]==8 || Alarm_Time[1]==10 || Alarm_Time[1]==12)//日越界判断
{
if(Alarm_Time[2]<1){Alarm_Time[2]=31;}//大月
if(Alarm_Time[2]>31){Alarm_Time[2]=1;}
}
else if(Alarm_Time[1]==4 || Alarm_Time[1]==6 || Alarm_Time[1]==9 || Alarm_Time[1]==11)
{
if(Alarm_Time[2]<1){Alarm_Time[2]=30;}//小月
if(Alarm_Time[2]>30){Alarm_Time[2]=1;}
}
else if(Alarm_Time[1]==2)
{
if(Alarm_Time[0]%4==0)
{
if(Alarm_Time[2]<1){Alarm_Time[2]=29;}//闰年2月
if(Alarm_Time[2]>29){Alarm_Time[2]=1;}
}
else
{
if(Alarm_Time[2]<1){Alarm_Time[2]=28;}//平年2月
if(Alarm_Time[2]>28){Alarm_Time[2]=1;}
}
}
if(Alarm_Time[3]<0){Alarm_Time[3]=23;}//时越界判断
if(Alarm_Time[4]<0){Alarm_Time[4]=59;}//分越界判断
if(Alarm_Time[5]<0){Alarm_Time[5]=59;}//秒越界判断
}
//更新显示,根据SetAlarmSelect和SetAlarmFlashFlag判断可完成闪烁功能
if(SetAlarmSelect==0 && SetAlarmFlashFlag==1){LCD_ShowString(1,1," ");}
else {LCD_ShowNum(1,1,Alarm_Time[0],2);}
if(SetAlarmSelect==1 && SetAlarmFlashFlag==1){LCD_ShowString(1,4," ");}
else {LCD_ShowNum(1,4,Alarm_Time[1],2);}
if(SetAlarmSelect==2 && SetAlarmFlashFlag==1){LCD_ShowString(1,7," ");}
else {LCD_ShowNum(1,7,Alarm_Time[2],2);}
if(SetAlarmSelect==3 && SetAlarmFlashFlag==1){LCD_ShowString(2,1," ");}
else {LCD_ShowNum(2,1,Alarm_Time[3],2);}
if(SetAlarmSelect==4 && SetAlarmFlashFlag==1){LCD_ShowString(2,4," ");}
else {LCD_ShowNum(2,4,Alarm_Time[4],2);}
if(SetAlarmSelect==5 && SetAlarmFlashFlag==1){LCD_ShowString(2,7," ");}
else {LCD_ShowNum(2,7,Alarm_Time[5],2);}
}
SetAlarm.h
#ifndef _SETALARM_H_
#define _SETALARM_H_
extern unsigned int SetAlarmSelect,SetAlarmFlashFlag;
extern int Alarm_Time[];//对应年、月、日、时、分、秒,用于记录闹钟响的时间
void SetAlarm(void);//闹钟响铃时间设置功能
#endif
工程代码及示例视频网盘链接:
[http://链接:pan.baidu.com/s/1wO8tebGC… 提取码:yyds]
初学单片机,有不少地方不太懂,还请多多指教。