感应开关盖垃圾桶项目(一)

定时/计时器

在之前的项目中,我们的延时操作都是借助软件延时,这样会占用CPU的资源导致开发效率较低,同时也可能会导致信号的丢失,因此引入定时/计时器的概念。
所谓延时,就要求我们进行“数数”的操作,而定时器是依靠内部震荡电路数数,计时器依靠外面的信号,读取针脚的数据。

定时器如何定时

本质原理:每经过一个机器周期就+1
在单片机中,机器周期通常是完成一个基本操作所需要的时间,它由若干个时钟周期组成,而时钟周期是CPU晶振工作频率的倒数。
在这里插入图片描述
在STC89C52系列单片机中,有两种计数速率:一种是12T模式,每12个时钟周期加1,与传统8051单片机相同。另外一种是6T模式,每6个时钟周期加1,速度是传统8051单片机的两倍
当晶振频率是11.0592MHz的时候,等于11059.2KHz = 11059200Hz
机器周期 = 12 x 时钟周期 =12 x (1/时钟频率) 秒 = 12 / 时钟频率 秒 = 12 / 11059200 秒 = 12 000 000
/ 11059200 微秒 = 1.085 微秒

定时器借助控制寄存器TCON

在这里插入图片描述

TCON格式
其中TF0为定时器/计时器T0溢出中断标志,TL0为低8位,TH0为高8位,一共可数16位即2^16=65536,当T0达到65536溢出时,会发出中断告知CPU已经数完了。最多数65536下约为71ms。
因此,当我们需要数10ms时,就需要从65536倒推,10ms=10000微秒,每个机器周期为1.085微秒,10ms占用10000/1.085=9216个机器周期,因此我们初始应定义为65536-9216=56320,将56320转换为16进制为DC00,因此低8位TL0为00,高八位TH0为DC
用软件生成验证:

void Timer0Init(void)		//10毫秒@11.0592MHz
{
	AUXR &= 0x7F;		//定时器时钟12T模式
	TMOD &= 0xF0;		//设置定时器模式
	TL0 = 0x00;		//设置定时初值
	TH0 = 0xDC;		//设置定时初值
	TF0 = 0;		//清除TF0标志
	TR0 = 1;		//定时器0开始计时
}

如何知道爆表溢出了?
通过TCON寄存器的bit5,当爆表时硬件会修改bit5位上面的数据,置1。可以通过中断清零,也可以通过代码清零
怎么样开始计时?
TR0位定时器的运行控制位,对应TCON的bit4,当赋值为1时就允许T0开始计时1,赋值为0时禁止T0计时
定时器使用时的多种模式
TMOD为定时器模式寄存器,可以通过修改TMOD的值改变定时器模式
在这里插入图片描述
了解了定时器的工作原理后,可以进行代码的开发。在这里我想要实现蜂鸣器500ms响一次灭一次,led灯每1000ms改变一次状态。
首先就是对定时器的计时模式进行配置,我希望用到16位的计时器,因此参照上图将M0,M1配置为0,1即TMOD=0x01;接下去定义一个10ms的计时时间,之后开始计时,将TR0置为1,依次循环查询检测是否爆表,用一个变量参数记录爆表的次数,当爆表次数达到50次说明过了500ms,使蜂鸣器响,当爆表次数达到100,说明过了1000ms,使小灯亮,同时需要注意变量的初始化与恢复。代码如下:

#include "reg52.h"

sbit led = P3^6;
sbit sounder = P2^5;

void main()
{
	int cnt = 0;//用来记录爆表的次数
	
	led = 1;
	sounder = 1;
	//1.配置定时器0工作模式为16位计时
	TMOD = 0x01;
	//2.给初值,定一个10ms出来
	TL0 = 0x00;
	TH0 = 0xDC;
	//3.开始计时
	TR0 = 1;
	TF0 = 0;
	//4.爆表了,操作led吗,累计到1s再操作led,每次爆表变量+1,变量到100再操作led
	while(1){
		if(TF0 == 1){	//爆表了
			TF0 = 0;	//软件清零
			cnt++;
			//重新给初值
			TL0 = 0x00;
			TH0 = 0xDC;
			if(cnt == 50){
				sounder = 0;
			}
			if(cnt == 100){
				cnt = 0;
				sounder = 1;
				led = !led;	//每经过1s翻转led的状态
			}
		}
	}
}
 

在这里插入图片描述

按位操作

因为TMOD有8位,像TMOD = 0x01;我们是希望使定时器0的M1和M0位修改位0和1,但是我们这样赋值并没有考虑定时器1的情况,如果定时器1也参与程序,这样赋值可能会导致定时器1的结果被我们修改,因此我们可以考虑按位操作。

&操作可以看成乘法运算
|操作可以看成加法运算
!操作就是取反

当我们需要对后四位(即定时器0)进行赋值时,可以先与0xF0进行与操作,这样可以使得后4位清0,但不会修改前四位的值。接下去与我们要修改的模式值即0x01进行或操作,这样就可以保证前四位不变,且按要求修改我们的后四位。封装后代码如下:

void delay10ms()
{
	//1.配置定时器0工作模式为16位计时
	//TMOD = 0x01;
	TMOD &= 0xF0;	//将低四位清零,且不改变高四位的值
	TMOD |= 0x01;	//将低四位修改为,且不改变高四位的值
	//2.给初值,定一个10ms出来
	TL0 = 0x00;
	TH0 = 0xDC;
	//3.开始计时
	TR0 = 1;
	TF0 = 0;
}

AUXR寄存器

可以用来降低单片机时钟对外界的电磁辐射
在这里插入图片描述
bit0置1时就可以实现禁止ALE信号输出。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值