51单片机入门_江协科技_17~18_OB记录的笔记

17. 定时器

  • 17.1.
    定时器介绍:51单片机的定时器属于单片机的内部资源,其电路的连接和运转均在单片机内部完成,无需占用CPU外围IO接口;

    定时器作用:
    (1)用于计时系统,可实现软件计时,或者使程序每隔一固定时间完成一项操作;
    (2)替代长时间的Delay,提高CPU的运行效率和处理速度,因为delay程序占用CPU资源,同时导致CPU无法处理外部诸如按键的操作;
    (…)

  • 17.2. STC89C52 定时器相关扩展阅读:STC89C52的介绍手册
    官方链接地址: https://www.stcmcu.com/datasheet/stc/STC-AD-PDF/STC89C51RC-RD+_GUIDE-CHINESE.pdf

  • 17.3. STC89C52定时器资源
    •定时器个数:3个(T0、T1、T2),T0和T1与传统的51单片机兼容,T2是此型号单片机增加的资源
    •注意:定时器的资源和单片机的型号是关联在一起的,不同的型号可能会有不同的定时器个数和操作方式,但一般来说,T0和T1的操作方式是所有51单片机所共有的

  • 17.4. 51单片机定时器内部工作原理
    •定时器在单片机内部就像一个小闹钟一样,根据时钟的输出信号,每隔“一秒”,计数单元的数值就增加一,当计数单元数值增加到“设定的闹钟提醒时间”时,计数单元就会向中断系统发出中断申请,产生“响铃提醒”,使程序跳转到中断服务函数中执行
    在这里插入图片描述

  • 17.5. 定时器工作模式:
    •STC89C52的T0和T1均有四种工作模式:
    模式0:13位定时器/计数器
    模式1:16位定时器/计数器(常用)
    模式2:8位自动重装模式
    模式3:两个8位计数器
    •工作模式1框图:
    在这里插入图片描述

    •SYSclk:系统时钟,即晶振周期,本开发板上的晶振为12MHz

  • 17.6. 中断系统

    • 中断系统是为使CPU具有对外界紧急事件的实时处理能力而设置的。
    • 当中央处理机CPU正在处理某件事的时候外界发生了紧急事件请求,要求CPU暂停当前的工作,转而去处理这个紧急事件,处理完以后,再回到原来被中断的地方,继续原来的工作,这样的过程称为中断。实现这种功能的部件称为中断系统,请示CPU中断的请求源称为中断源。微型机的中断系统一般允许多个中断源,当几个中断源同时向CPU请求中断,要求为它服务的时候,这就存在CPU优先响应哪一个中断源请求的问题。通常根据中断源的轻重缓急排队,优先处理最紧急事件的中断请求源,即规定每一个中断源有一个优先级别。CPU总是先响应优先级别最高的中断请求。
    • 当CPU正在处理一个中断源请求的时候(执行相应的中断服务程序),发生了另外一个优先级比它还高的中断源请求。如果CPU能够暂停对原来中断源的服务程序,转而去处理优先级更高的中断请求源,处理完以后,再回到原低级中断服务程序,这样的过程称为中断嵌套。这样的中断系统称为多级中断系统,没有中断嵌套功能的中断系统称为单级中断系统。
  • 17.7. 中断程序流程:
    在这里插入图片描述

  • 17.8. 中断资源:
    •中断源个数:8个(外部中断0、定时器0中断、外部中断1、定时器1中断、串口中断、定时器2中断、外部中断2、外部中断3)
    •中断优先级个数:4个
    •中断号:
    在这里插入图片描述

•注意:中断的资源和单片机的型号是关联在一起的,不同的型号可能会有不同的中断资源,例如中断源个数不同、中断优先级个数不同等等

  • 17.9. 定时器和中断系统;
    在这里插入图片描述

  • 17.10. 定时器相关寄存器:

在这里插入图片描述

18. 按键控制LED流水灯模式&定时器时钟

  • 18.1. 计时器子函数与中断子函数测试程序:
#include <REGX52.h>

void TimerR0_Init() //定时器初始化子函数
{
	//TMOD=0x01; //0000 0001;如果同时使用T0与T1则此TMOD赋值方式有问题
	//TMOD=TMOD&0xf0;//把TMOD低4位清0,高4位不变;与清0;
	//TMOD=TMOD|0x01;//把TMOD最低位置1,高4位不变;或置1;
	TMOD&=0xf0;//上述简写
	TMOD|=0x01;//上述简写
	TF0=0;
	TR0=1;
	TH0=64535/256; //高8位数值,从64535到65535需要1000us,超出65535后溢出
	TL0=64535%256; //低8位数值
	ET0=1;
	EA=1;
	PT0=0;
}

void main() //主程序
{
	TimerR0_Init();
	while(1)
	{
		
	}
	
}

unsigned int T0_Count;
void TimerR0_Routine() interrupt 1 //中断子函数
{
	TH0=64535/256; //重新赋予初值,防止从0开始计数
	TL0=64535%256; //重新赋予初值,防止从0开始计数	
	T0_Count++;
	if(T0_Count>=1000)
	{
		T0_Count=0;
		P2_0=~P2_0;
	}
}
  • 18.2. STC-ISP中定时器的相关配置
    上述程序与STC-ISP中的示例定时器计算器中的程序有1us的差距,同时需要注意STC-ISP中的软件的配置:
    在这里插入图片描述

此时的C代码为:

void Timer0_Init(void)		//1毫秒@12.000MHz
{
	//AUXR &= 0x7F;			//定时器时钟12T模式,新版本问题此语句删除
	TMOD &= 0xF0;			//设置定时器模式
	TMOD |= 0x01;			//设置定时器模式
	TL0 = 0x18;				//设置定时初始值 计算后与64535%256=17不同;
	TH0 = 0xFC;				//设置定时初始值 计算后与64535/256相同; 
	TF0 = 0;				//清除TF0标志
	TR0 = 1;				//定时器0开始计时
}

  • 18.3. 示例18.1中的程序更新后如下:
#include <REGX52.h>

void TimerR0_Init() //定时器0初始化子函数
{
	
	TMOD&=0xf0;//上述简写
	TMOD|=0x01;//上述简写
	TF0=0;
	TR0=1;
	TH0=0xfc; //高8位数值,从64535到65535需要1000us,超出65535后溢出
	TL0=0x18; //低8位数值
	ET0=1;
	EA=1;
	PT0=0;
}

void main() //主程序
{
	TimerR0_Init();
	while(1)
	{
		
	}
	
}

unsigned int T0_Count;
void TimerR0_Routine() interrupt 1 //中断子函数
{
	TH0=64535/256; //重新赋予初值,防止从0开始计数
	TL0=64535%256; //重新赋予初值,防止从0开始计数	
	T0_Count++;
	if(T0_Count>=1000)
	{
		T0_Count=0;
		P2_0=~P2_0;
	}
}
  • 18.4. 定时器函数模块化(模块化定时器的初始化程序)
  • TimeR0.c
#include <REGX52.H>
/**
  * @brief 定时器初始化子函数,1ms,12MHz
  * @param 无
  * @retval 无
  */


void TimeR0_Init() //定时器初始化子函数
{
	//AUXR &= 0x7F;			//定时器时钟12T模式
	TMOD &= 0xF0;			//设置定时器模式
	TMOD |= 0x01;			//设置定时器模式
	TL0 = 0x18;				//设置定时初始值
	TH0 = 0xFC;				//设置定时初始值
	TF0 = 0;				//清除TF0标志
	TR0 = 1;				//定时器0开始计时
	ET0=1;
	EA=1;  //
	PT0=0; //中断优先级
}

//以下程序全部注释方便后续调用,定时器中断函数模板
//
//void TimeR0_Routine() interrupt 1 //中断子函数
//{
//	static unsigned int T0Count //设置静态子函数用T0Count,防止T0Count丢失
//	TL0=0x18;				//设置定时初始值
//	TH0=0xFC;				//设置定时初始值
//	T0Count++;
//	if(T0Count>=1000)
//	{
//		T0Count=0;
//		
//	}
//}
  • TimeR0.h
#ifndef _TIMER0_H_ 
#define _TIMER0_H_  

void TimeR0_Init();

#endif
  • 18.5. 4个独立按键的模块化程序:
  • Four_Key.c
#include <REGX52.h>
#include "delay_xms.h"
/**
  * @brief 4个独立按键按下判断程序,按键松手返回KeyNum
  * @param 无
  * @retval KeyNum 的返回值,0~4,无按键按下时返回值为0
  */



unsigned char Four_Key()
{
	unsigned char KeyNum=0;
	
		if(P3_1==0){delay_xms(20);while(P3_1==0);delay_xms(20);KeyNum=1;}
		if(P3_0==0){delay_xms(20);while(P3_0==0);delay_xms(20);KeyNum=2;}
		if(P3_2==0){delay_xms(20);while(P3_2==0);delay_xms(20);KeyNum=3;}
		if(P3_3==0){delay_xms(20);while(P3_3==0);delay_xms(20);KeyNum=4;}
	
	return KeyNum;
}


  • Four_Key.h
#ifndef __H_ 
#define __H_  

 unsigned char Four_Key();

#endif
  • 针对建立好的模块化程序进行简单测试,按下相应按键,P2的相应的LED灯亮起;
#include <REGX52.h>
#include "TimeR0.h"
#include "Four_Key.h"

unsigned char keynum;
void main() //主程序
{
	TimeR0_Init();
	
	while(1)
	{
		keynum=Four_Key();
		if(keynum)
		{
		if (keynum==1){P2_0=!P2_0;}
		if (keynum==2){P2_1=!P2_1;}
		if (keynum==3){P2_2=!P2_2;}
		if (keynum==4){P2_3=!P2_3;}		
	    }
	}
	
}


void TimerR0_Routine() interrupt 1 //中断子函数
{
	static unsigned int T0Count; //设置静态子函数用T0Count,防止T0Count丢失
	TL0=0x18;				//设置定时初始值
	TH0=0xFC;				//设置定时初始值
	T0Count++;
	if(T0Count>=1000)
	{
		T0Count=0;
		
	}
}
  • 测试程序Proteus中测试没有问题,按键按一下相应LED灯亮起,再按一下熄灭;同时测试了独立按键检测程序和中断函数模块程序没有问题。
    在这里插入图片描述

  • 18.6. LED流水灯控制的实现

  • 通过包含 INTRINS.h的函数库中的_cror_与_crol_循环左移与循环右移函数来实现;

#include <REGX52.h>
#include "TimeR0.h"
#include "Four_Key.h"
#include <INTRINS.h>

unsigned char keynum,LED_Mode; //增加LED_Mode变量控制LED灯模式

void main() //主程序
{
	TimeR0_Init();
	P2=0xfe; //P2口LED灯赋予初始值,保证至少1个LED灯亮起;
	while(1)
	{
		keynum=Four_Key();
		if(keynum)
		{
			if (keynum==1) //当S1按下松开后
			{
				LED_Mode++; //LED_Mode数值自动更新
			if (LED_Mode>=2) 
				LED_Mode=0;
			
			}
		}
	}
	
}


void TimerR0_Routine() interrupt 1 //中断子函数
{
	static unsigned int T0Count; //设置静态子函数用T0Count,防止T0Count丢失
	TL0=0x18;				//设置定时初始值
	TH0=0xFC;				//设置定时初始值
	T0Count++;
	if(T0Count>=500)
	{
		T0Count=0;
		if (LED_Mode==0)
			P2=_crol_(P2,1);//通过_crol_(P2,1)函数控制LED左移
		if (LED_Mode==1)
			P2=_cror_(P2,1);//通过_cror_(P2,1)函数控制LED右移
	}
}
  • 18.7. 通过LCD1602来制作一个定时器时钟程序
  • 定时器时钟程序如下:
#include <REGX52.h>
#include "LCD1602.h"
#include "TimeR0.h"

unsigned char sec=55,min=59,hour=23;
void main()
{
	TimeR0_Init();
	LCD_Init();
	LCD_ShowString(1,1,"Clock:");//1行1列显示Clock:
	LCD_ShowString(2,1,"  :  :");//2行中需要显示的分割冒号
	while(1)
	{
		
		LCD_ShowNum(2,7,sec,2);//秒显示
		LCD_ShowNum(2,4,min,2);//分钟显示
		LCD_ShowNum(2,1,hour,2);//小时显示
		
	}
	
}

void TimeR0_Routine() interrupt 1 //中断子函数
{
	static unsigned int T0Count; //设置静态子函数用T0Count,防止T0Count丢失
	TL0=0x18;				//设置定时初始值
	TH0=0xfc;				//设置定时初始值
	T0Count++;
	if(T0Count>=1000)
	{
		T0Count=0;
		sec++;
		if(sec>=60)
		{
			sec=0;
			min++;
			if(min>=60)
			{
				min=0;
				hour++;
				if(hour>=24)
				{
					hour=0;
				}
			}
		}
	}
		
	
}
  • proteus测试结果没问题;
    在这里插入图片描述

  • 7
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值