51 单片机 (3) 定时器/计数器 之 利用定时器0(timer0)编写精确的延时函数

定时器/计数器

什么是定时器/计数器?

在51单片机中,定时器/计数器是用来实现定时功能,并且具有计数的功能,来实现对外部信号的计数,其实他们是同一个物理的电子元件。
定时器实际上也是工作在计数方式下,只是计数的是固定周期的脉冲,由于脉冲周期固定,由计数值可以计算时间,有定时功能。
定时和计数只是触发来源不同(时钟信号和外部脉冲)其他方面是一样的。

AT89C51的定时器/计数器

在这里插入图片描述
从上图我们可以发现这款51单片机有2个16位的定时/计数器,他们被标识为T0和T1。

定时器/计数器相关的寄存器

在这里插入图片描述

TCON

T0、T1定时器/计数器控制寄存器,格式如下
在这里插入图片描述

TF0、TF1

定时器/计数器T0(T1)溢出标志。当T0(T1)被允许计数以后,从初值开始加1计数。当最高位产生溢出时由硬件对TF0(TF1)置1,中断完成又由硬件对TF0(TF1)清0

TR0、TR1

定时器T0(T1)的运行控制位。该位由软件置位和清零。
对T0
当GATE (TMOD.3) =0,TR0=1时就允许T0开始计数
当GATE (TMOD.3) =0,TR0=0时就禁止T0计数
当GATE (TMOD.3) =1,TR0=1且INT0输入高电平时,才允许T0计数。
对T0
当GATE (TMOD.7) =0,TR1=1时就允许T1开始计数
当GATE (TMOD.7) =0,TR1=0时就禁止T1计数
当GATE (TMOD.7) =1,TR1=1且INT1输入高电平时,才允许T1计数。

TMOD

TMOD是定时器、计数器模式控制寄存器
在这里插入图片描述使用定时器0就使用第四位,定时器1就用高四位

GATE

门控位。GATE=0,以运行控制位TR启动定时器;GATE=1,以外中断请求信号(INT1或INT0)启动定时器,这可以用于外部脉冲宽度测量。在TMOD中GATE一般情况下都等于0。

C/T

控制其用作定时器还是计数器
置0用作定时器(从内部系统时钟输入)
置1用作计数器(从T0/P3.4 或T1/P3.5脚输入)

M1、M0

定时器/计数器模式选择
M1----M0-----------功能
0-------0-------13位定时器/计数器,TL用低5位,TH全用
0-------1-------16位定时器/计数器,TL、TH全用
1-------0--------8位自动重装定时器,当溢出时TH的值自动重装入TL
1-------1--------定时器/计数器无效(停止计数)

TH、TL

定时器值的存储寄存器

名称描述
TH0定时器0高字节
TL0定时器0低字节
TH1定时器1高字节
TL1定时器1低字节

实验内容

利用AT89C51的定时器0来实现精确延时

实验环境

  • 仿真软件
    Protue 8.9 sp2
  • IDE
    Keil5 C51
  • 单片机
    AT89C51

Protues仿真图

在这里插入图片描述注意,这里我们使用12M的时钟频率
在这里插入图片描述

Keil工程

项目结构

在这里插入图片描述

代码

#include <reg52.h>
//时钟频率
#define FOSC 12000000L
//计算器初值计算
#define Times (65536 - FOSC / 12 / 1000)
//LED1控制引脚
sbit led1 = P1 ^ 0;

//计数器中断次数
volatile unsigned int count;

//定时器溢出中断(1ms中断一次)
void Timer0_Rountine(void) interrupt 1
{
    //重新装载初值
    TL0 = Times;
    TH0 = Times >> 8;
    //总延时减1
    count--;
}

//毫秒级延时
void delay_ms(unsigned int ms)
{
    //给T0低字节装载初始值
    TL0 = Times;
    //给T0高字节装载初始值
    TH0 = Times >> 8;

    //初始化T0模式寄存器,也就是TMOD的第四位
    TMOD &= 0xF0;
    //取值为0001
    //即GATE=0
    //C/T=0 我们用作定时器
    //M1=0 M2=1 模式选择为16位的定时器
    TMOD |= 0x01; //xxxx 0001

    //让计数器开始计数
    TR0 = 1;
    //打开定时器0的中断开关
    ET0 = 1;
    //打开中断总开关
    EA = 1;

    //延时count毫秒
    count = ms;

    //当cout等于0是,关闭计数,关闭T0的中断
    while (count > 0)
        ;
    {
        TR0 = 0;
        ET0 = 0;
    }
}

void main(void)
{
    while (1)
    {
        led1 = 1;
        delay_ms(500);
        led1 = 0;
        delay_ms(500);
    }
}

结果

在这里插入图片描述

总结

最后我想说一下关于时间的计算

时钟周期

一个CPU周期时间又包含若干个时钟周期。时钟周期定义为时钟脉冲的倒数(可以这样来理解,时钟周期就是单片机外接晶振的倒数,例如12M的晶振,它的时间周期就是1/12 μs),是计算机中最基本的、最小的时间单位。在一个时钟周期内,CPU仅完成一个最基本的动作。由于时钟脉冲是计算机的基本工作脉冲,它控制着计算机的工作节奏(使计算机的每一步都统一到它的步调上来)。显然,对同一种机型的计算机,时钟频率越高,计算机的工作速度就越快。但是,由于不同的计算机硬件电路和器件的不完全相同,所以其所需要的时钟周频率范围也不一定相同。我们学习的 8051单片机的时钟范围是1.2MHz-12MHz。
一个机器周期包含六个状态周期(用S表示)。一个状态周期有两个节拍(用P1、P2表示)。8051系列单片机的一个机器周期同6 个S周期(状态周期)组成。也就是说一个机器周期=6个状态周期=12个振荡周期(即时钟周期)。

机器周期

在计算机中,为了便于管理,常把一条指令的执行过程划分为若干个阶段,每一阶段完成一项工作。例如,取指令、存储器读、存储器写等,这每一项工作称为一个基本操作。完成一个基本操作所需要的时间称为机器周期。一般情况下,一个机器周期由若干个S周期(状态周期)组成。通常用内存中读取一个指令字的最短时间来规定CPU周期,(也就是计算机通过内部或外部总线进行一次信息传输从而完成一个或几个微操作所需要的时间)),它一般由12个时钟周期(振荡周期)组成,也是由6个状态周期组成。而振荡周期=1秒/晶振频率,因此单片机的机器周期=12秒/晶振频率 。

所以在51 芯片中定时器启动后会在每一个机器周期会使定时器值的存储寄存器增加一,一个机器周期等于十二个振荡周期,所以可以得知51芯片的计数速率为晶体振荡器频率的1/12,一个12M 的晶振用在51芯片上,那么51的计数速率就为1M。
即一秒钟1000000次,1ms=1000次,我们用的是16位的定时器,则可以存储2^16=65536次,所用根本用不玩,于是我们就从65536-1000开始计数就完美解决了,下面这个初值就是这样算出来的

//时钟频率
#define FOSC 12000000L
//计算器初值计算
#define Times (65536 - FOSC / 12 / 1000)

同理若为11.0592M计算为

//时钟频率
#define FOSC 11059200L
//计算器初值计算
#define Times (65536 - FOSC / 12 / 1000)
好的,以下是一个基于51单片机的程序,使用定时器/计数器中断实现秒闪烁和分闪烁的效果,LED灯分别连接在P1口和P2口上: ```C #include <reg51.h> #define FREQ 11059200L // 定义晶振频率 #define TIMER_MS 10 // 定义定时器中断时间间隔(毫秒) #define LED_SEC P1 // 定义秒LED所在的端口 #define LED_MIN P2 // 定义分LED所在的端口 unsigned int timer_count = 0; // 定义计数器变量,用于记录定时器中断次数 // 定时器中断服务函数 void Timer0_ISR() interrupt 1 { TH0 = (65536 - (FREQ / 12 / 1000)); // 重新设置计时器初值 TL0 = (65536 - (FREQ / 12 / 1000)) % 256; timer_count++; // 计数器加1 } // 延时函数,用于控制LED闪烁的时间间隔 void Delay(unsigned int time) { unsigned int i, j; for (i = time; i > 0; i--) { for (j = 0; j < 120; j++); } } void main() { unsigned int sec_flag = 0; // 秒闪烁标志位 unsigned int min_flag = 0; // 分闪烁标志位 TMOD = 0x01; // 设置定时器0为模式1(16位定时器) TH0 = (65536 - (FREQ / 12 / 1000)); // 设置计时器初值 TL0 = (65536 - (FREQ / 12 / 1000)) % 256; EA = 1; // 开启总中断 ET0 = 1; // 开启定时器0中断 TR0 = 1; // 开启定时器0 while (1) { if (timer_count * TIMER_MS >= 1000) { // 判断是否达到1秒 timer_count = 0; // 计数器清零 sec_flag = !sec_flag; // 秒闪烁标志位反转 if (sec_flag) { LED_SEC |= 0x01; // 点亮秒LED } else { LED_SEC &= 0xFE; // 熄灭秒LED } } if (timer_count * TIMER_MS >= 60000) { // 判断是否达到1分钟 min_flag = !min_flag; // 分闪烁标志位反转 timer_count = 0; // 计数器清零 if (min_flag) { LED_MIN |= 0x01; // 点亮分LED } else { LED_MIN &= 0xFE; // 熄灭分LED } } Delay(10); // 延时10毫秒 } } ``` 这份代码使用了定时器/计数器中断来实现LED的闪烁效果,当中断次数达到一定值时,就判断是否达到需要闪烁的时间间隔,来控制LED的状态变化。需要注意的是,定时器频率的设置需要根据实际的晶振频率进行调整。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值