51单片机笔记:定时器/计数器

单片机笔记

定时器/计数器

定时器/计数器的结构

AT89S51内部两个16位定时器/计数器:T0(P3.4),T1(P3.5),定时器/计数器T0由特殊寄存器TH0,TL0构成,T1由特殊功能寄存器TH1,TL1构成

T0,T1都有定时器和计数器两种工作模式,两种模式实质都是对脉冲信号进行计数,只不过技术信号来源不同。

  • 计数器模式 是对T0(P3.4)和T1(P3.5)两个引脚上的外部脉冲进行计数

  • 定时器模式是对系统时钟信号(fosc)经12分频后的内部脉冲信号(机器周期Tcy)计数.由于系统时钟频率fosc是定值,可根据数值计算出定时时间T(定时时间T = n * Tcy).

振荡周期Tosc = 1/fosc, 机器周期Tcy = 12Tosc = 12/fosc)

例如:fosc = 12Hz, Tcy = 1us

T0,T1属于加1计数器,即每记一个脉冲,计数器加1.(51单片机)

T0,T1具有4种工作方式(方式0,1,2,3)

特殊功能寄存器TMOD用于选择定时器/计数器T0,T1的工作模式和工作方式.

特殊功能寄存器TCON用于控制T0,T1的启动和停止计数,同时包含了T0,T1状态.

计数器起始计数从初值开始,单片机复位时计数器初值为0,也可给计数器装入一个新的初值(0 ~ 2^16 - 1).

若计数器溢出会产生中断,称溢出中断,最大计数值2^16 = 65536, 此时刻发生溢出中断, 最大初值2^16 - 1 = 65535

工作方式控制寄存器TMOD

寄存器地址89H, 不可按位操作, 高4位控制T1, 低4位控制T0.

D7D6D5D4D3D2D1D0
GATEC/T’M1M0GATEC/T’M1M0

D7-D4 : T1方式字段, D3-D0 : T0方式字段

M1, M0工作方式选择位
M1M0工作方式
00方式0, 13位定时器/计数器
01方式1, 16位定时器/计数器
10方式2, 8位的常数自动重新装载的定时器/计数器
11方式3, 仅适用于T0, 此时T0分成2个8位计数器, T1停止计数

方式0为兼容早期8048的13位定时器/计数器

GATE - 门控位
  • GATE = 0, 定时器是否计数, 由启动控制位TRx(x = 0, 1)来控制. (TRx = 1 时启动)
  • GATE = 1, 定时器是否计数, 由外部中断引脚INTx’上的电平与运行控制位TRx共同控制.(1 == INTx’ & 1 == TRx 为真时启动)
C/T’ - 计数器模式和定时器模式选择位

**C : Counter, T : Timer **

  • C/T’ = 0, 定时器模式, 对系统时钟12分频后的脉冲(fosc/12)进行计数.
  • C/T’ = 1, 计数器模式, 对计数器外部输入引脚T0(P3.4)或T1(P3.5)的外部脉冲(负跳变, 即对下降沿计数)计数.
定时器/计数器控制寄存器TCON

TCON字节地址88H, 位地址为88H~8FH.

D7D6D5D4D3D2D1D0
TF1TR1TF0TR0IE1IT1IE0IT0
TF1, TF0 - 计数溢出标志位

当计数器计数溢出时, 该位置1. 使用查询方式时, 此位可提供CPU查询, 但应注意查询后, 用软件及时该位清0. 使用中断方式时, 作为中断请求标志位, 进入终端服务程序后由硬件自动清0.

TR1, TR0 - 计数运行控制位
  • TR1/TR0位 = 1, 启动计数器计数的必要条件.
  • TR1/TR0位 = 0, 停止计数器计数.

该位可由软件置1和清0

定时器/计数器的4种工作方式

方式1(M1 = 0, M0 = 1)

定时器计数器方式1的逻辑结构框图

方式1和方式0差别仅存在于计数器的位数不同, 方式1为16位计数器, 由THx高8位和TLx低8位构成(x = 0,1), 方式0为13位计数器, 有关控制状态位含义(GATE, C/T’, TFx, TRx)与方式0相同.

方式0(M1 = 0, M0 = 0)

定时器计数器方式0的逻辑结构框图

该方式为13位定时/计数器的计数最大值位2^13 = 8196

方式2

方式0和方式1最大特点是计数溢出后, 计数器为全0. 因此在循环定时或循环计数应用时就存在用指令反复装入计数初值的问题, 这会影响定时精度(指令本身执行也需要时间), 方式2就是为解决此问题而设置的.

M1, M0 = 1, 0时, 工作方式2. 8位自动重装载方式

方式2的逻辑结构框图

低8位用来计数/定时, 高8位用来保存初值, 硬件自动完成装入初值的操作

最大计数/定时范围2^8 = 256

方式3

方式3是为增加一个附加的8位定时器/计数器而设置的, 从而使AT89S51具有3个定时器/计数器. 方式3只适用于T0, T1不能工作在方式3
T0在方式3下, 分为了两个8位定时器/计数器

方式3的逻辑结构框图

对外部输入的计数信号的要求

计数器模式时, 计数脉冲来自外部输入引脚T0或T1. 当输入信号产生负跳变时, 计数值增1.

由于确认一次负跳变要花2个机器周期, 即24个振荡周期, 因此外部输入的计数脉冲的最高频率为系统振荡器频率的1/24

对外输入信号占空比没有限制, 但为确保某一给定电平在变化前能被采样1次, 该电平至少保持1个机器周期.

方式1应用 : P1口控制8只LED每0.5s闪亮一次
分析

设置晶振频率12MHz, 则机器周期为1us, 一次溢出中断时间的总耗时计算65536 * 1us = 65…ms

题目要求0.5s即500ms, 500ms > 65ms,故需软件计数, 500ms约溢出中断次数为100次

可用软件编写计时器对中断次数进行计数

设置TMOD寄存器
  1. TMOD寄存器的M1 = 0, M0 = 1则T0工作在方式1
  2. 设置C/T’ = 0, 为定时器模式
  3. GATE为0, 对T0的运行控制仅由TR0来控制
  4. 定时器T1不使用, 各相关位均设为0
D7D6D5D4D3D2D1D0
GATEC/T’M1M0GATEC/T’M1M0
00000001

TMOD初始化值则为0x01

计算定时器T0的计数初值

设定时时间5ms(即5,000us), 设T0计数初值为X, 假设晶振频率为11.0592MHz, 则定时时间为:

定时时间 = (2^16 - X) * Tcy = (2^16 - X) * 12/晶振频率

则 5000 = (2^16 - X) * 12 / 11.0592, 得X = 60,928

转换成16进制为0xee00, 其中0xee装入TH0, 0x00装入TL0.

X --> 65536(产生溢出)

个数n = 65536 - X

定时时间T = n * Tcy == 5ms == 5,000us

机器周期Tcy =  12 / 11.0592 us
设置IE寄存器

本例采用定时器T0中断, 因此需将IE寄存器中的EA,ET0位置1

启动T0

将定时器控制寄存器TCON中的TR0置1, 则启动定时器T0

方式1定时中断控制LED闪亮的仿真图

源码
/* 
 * 功能实现: 使用方式1定时中断控制LED闪亮
 * 编写环境: Keil5
 * 硬件仿真: Proteus 8 Professional
 * 日期: 2022-11-05
 */

#include <reg51.h>

char i = 100; // 用于软件计数

int main(void)
{
    TMOD = 0x01; // 定时器T0为方式1
    TH0 = 0xee; // 设置定时器初值
    TL0 = 0x00;
    P1 = 0x00; // P1口8个LED点亮
    EA = 1; // 总中断开
    ET0 = 1; // T0中断开
    TR0 = 1; // 启动T0

    while(1) // 循环等待
    {
        ;
    }

    return 0;
}

void timer0() interrupt 1 // T0中断程序
{
    TH0 = 0xee; //重新赋初值
    TL0 = 0x00;

    // 软件计数, 100次溢出中断后, LED状态进行转换	
    i--; // 循环次数减1
    if (i <= 0)
    {
        P1 = ~P1; // P1口按位取反, LED状态转换
	i = 100; // 重置循环次数
    }

    return;
}
计数器应用

如图,T1采用计数模式,方式1中断,计数输入引脚T1(P3.5)上外接按钮开关,作为计数信号输入。按4次按钮开关后,P1口的8只LED闪烁不停。

计数器应用硬件仿真图

设置TMOD寄存器
  1. T1工作在方式1,应使TMOD的M1,M0 = 0,1
  2. 设置C/T’=1,为计数器模式
  3. 对T0运行控制仅由TR0来控制启动,应使GATE0=0.
  4. 定时器T0不使用,各相关位均设0.

所以, TMOD寄存器应初始化为0x50.

计算定时器T1的计数初值

由于每按1次按钮开关,计数1次,按4次后,P1口的8只LED闪烁不停. 因此计数器初值为65536 - 4 = 65532, 所以, TH1 = (65536 - 4) / 256, TL1 = (65536 - 4) % 256.

设置IE寄存器

由于采用T1中断,因此需将IE寄存器的EA, ET1位置1.

启动和停止定时器T1

将寄存器TCON中TR1 = 1, 则启动T1计数.

源码
#include <reg51.h>

void Delay(unsigned int i);

int main(void)
{
    TMOD = 0x50;	// 设置定时器T1位方式1计数
    TH1 = (65536 - 4) / 256;	// 向TH1写入初值的高8位
    TL1 = (65536 - 4) % 256;	// 向TL1写入初值的低8位
    EA = 1;	// 总中断允许
    ET1 = 1;	// 定时器T1中断允许
    TR1 = 1;	// 启动定时器T1

    while (1)
    {
        ;
    }

    return 0;
}

void Delay(unsigned int i)
{
    unsigned int j;
    for ( ; i > 0; i--)
    {
        for (j = 0; j < 125; j++)
	{
	    ;
	}
    }

    return;
}

void T1_int(void) interrupt 3 // T0中断时interrupt为0, T1中断时interrupt为3
{
    while (1)	// 一般中断函数不会设置为死循环
    {
    	// 单次的计数中断所以不需要重新赋初值
        P1 = 0xff;	// 8位LED全灭
	Delay(500);	// 延时500ms
	P1 = 0;		// 8位LED全亮
	Delay(500);	// 延时500ms
    }

    return;
}

习题

习题7-1

采用T0方式1的定时中断方式, 控制AT89C51的P1.1引脚输出100Hz的方波.

分析

f = 100Hz, T = 10ms, 定时时间为10/2 = 5ms(单高电平或低电平的时间) < 65ms, 所以定时时间T = 5ms

假设时钟频率fosc = 12MHz, 工作在方式1时最大时间为65536ms

TMOD = 0x01

5ms = 5000us = (2^16 - X) * 12 / 12, 解得X = 60,536

转换为16进制, 则初值X = 0xEC78, TH0 = 0xEC, TL0 = 0x78

因操作为P1.1需提前定义,如下

sbit Pulse =  P1^1;

// LED状态转换
Pulse = !Pulse;
源码
/* 
 * 功能实现: 采用T0方式1的定时中断方式, 控制AT89C51的P1.1引脚输出100Hz的方波.
 * 编写环境: Neovim + Keil5
 * 硬件仿真: Proteus 8 Professional
 * 日期: 2022-11-05
 */

#include <reg51.h>

sbit Pulse = P1^1; // 定义P1.1口为Pulse

int main(void)
{
    TMOD = 0x01; // 定时器T0为方式1

    TH0 = 0xEC; // 设置定时器初值
    TL0 = 0x78;
    /*
     * 计算过程: 
     * f = 100Hz, T = 10ms, 定时时间为10/2 = 5ms(单高电平或低电平的时间) < 65ms, 所以定时时间T = 5ms
     * 假设时钟频率fosc = 12MHz, 工作在方式1时最大时间为65536ms
     * 5ms = 5000us = (2^16 - X) * 12 / 12, 解得X = 60,536
     * 转换为16进制, 则初值X = 0xEC78, TH0 = 0xEC, TL0 = 0x78
     * TH0 = (65536-5000)/256, TL0 = (65536-5000)%256
     */

    Pluse = 0; // P1.1输出低电平
    EA = 1; // 总中断开
    ET0 = 1; // T0中断开
    TR0 = 1; // 启动T0

    while(1) // 循环等待
    {
        ;
    }

    return 0;
}

void timer0() interrupt 1 // T0中断程序
{
    TH0 = 0xEC; //重新赋初值, 循环计时的时候需要重新赋初值
    TL0 = 0x78;

    P1 = ~P1; // P1口按位取反, LED状态转换
    Pluse = !Pluse; // P1.1口按位取反, 高低电平转换

    return;
}
习题7-2

采用T0方式1的定时中断方式, 控制AT89C51的P1.1引脚输出100Hz, 占空比为20%的脉冲.

分析

高电平时间1/5 * T, T = 1/100 = 10ms; 所以高电平持续时间为2ms, 低电平持续时间为8ms.

定时器定时2ms, 设计软件计数器在中断一次后计4次中断(即8ms
)后置低电平

TMOD = 0x01;
/*
 * 2ms = 1us * 2000, 故n = 2000
 * TH0 = (2^16 - n) / 256
 */
TH0 = (65536 - 2000) / 256;
TL0 = (65536 - 2000) % 256;
源码
/* 
 * 功能实现: 采用T0方式1的定时中断方式, 控制AT89C51的P1.1引脚输出100Hz, 占空比为20%的脉冲
 * 编写环境: Neovim + Keil5
 * 日期: 2022-11-15
 */

#include <reg51.h>

sbit Pulse = P1^1; // 定义P1.1口为Pulse

unsigned char count = 0;

int main(void)
{
    TMOD = 0x01; // 定时器T0为方式1

    TH0 = (65536 - 2000) / 256; // 设置定时器初值
    TL0 = (65536 - 2000) % 256;

    Pluse = 0; // P1.1输出低电平
    EA = 1; // 总中断开
    ET0 = 1; // T0中断开
    TR0 = 1; // 启动T0

    while(1) // 循环等待
    {
        ;
    }

    return 0;
}

void timer0() interrupt 1 // T0中断程序
{

    TH0 = (65536 - 2000) / 256; //重新赋初值
    TL0 = (65536 - 2000) % 256;

    count++;

    if (1 == count) // 2ms过后, 高电平持续时间结束, 需将P1.1变为低电平
    {
        Pluse = 0;
    }
    else if (5 == count) // (5-1) * 2ms过后, 低电平持续时间结束, 需将P1.1变为高电平
    {
        Pluse = 1;
	count = 0; // 一个脉冲周期结束, 将count清零 
    }

    return;
}
课后题7-5

Proteus虚拟仿真. 使用T0, 采用方式2定时中断, 在P1.0引脚上输出周期为400us, 占空比为25%的矩形脉冲, 要求在P1.0引脚上接有虚拟示波器, 观察P1.0引脚输出的矩形脉冲波形.

分析

TMOD = 0000 0010

  • D0 = 0
  • D1 = 1 D1D0 = 10 为方式2
  • D3(C/T’) = 1, Counter计数器模式, 对计数器外部输入引荐T0(P3.4)或T1(P3.5)的外部脉冲(负跳变)计数
  • D3(GATE) = 0, 定时器是否计数, 由启动控制位TR0/TR1来控制(置1时启动)

高电平持续时间为1/4 * 400us = 100us, 低电平持续时间为300us

中断定时为100us, 设计软件计数器, 中断次数为1时跳变为低电平, 中断次数为(4 - 1)时, 跳变为高电平, 循环往复

12MHz时:

n = 100us / 1us = 100

TL0 = 2^8 - 100, TL1 = 2^8 - 100

方式2为8位, 且TH0用于储存初值, 故TL0和TH0二者数值相同, 不存在将十六进制数分位的问题

课后题7-5硬件仿真原理图
课后题7-5波形图

源码
/* 
 * 功能实现:  使用T0, 采用方式2定时中断, 在P1.0引脚上输出周期为400us, 占空比为25%的矩形脉冲 
 * 编写环境: Neovim + Keil5
 * 硬件仿真: Proteus 8 Professional
 * 日期: 2022-11-15
 */

#include <reg51.h>

sbit Pluse = P1^0; // 定义P1.0口为Pulse

unsigned char count = 0;

int main(void)
{
    TMOD = 0x01; // 定时器T0为方式1

    TH0 = 256 - 100; // 设置定时器初值
    TL0 = 256 - 100; 

    Pluse = 0; // P1.1输出低电平
    EA = 1; // 总中断开
    ET0 = 1; // T0中断开
    TR0 = 1; // 启动T0

    while(1) // 循环等待
    {
        ;
    }

    return 0;
}

void timer0() interrupt 1 // T0中断程序
{

    TH0 = 256 - 100; // 重新赋值
    TL0 = 256 - 100; 

    count++;

    if (1 == count) // 100us过后, 高电平持续时间结束, 需将P1.1变为低电平
    {
        Pluse = 0;
    }
    else if (4 == count) // (4-1) * 100us过后, 低电平持续时间结束, 需将P1.1变为高电平
    {
        Pluse = 1;
		count = 0; // 一个脉冲周期结束, 将count清零 
    }

    return;
}
  • 15
    点赞
  • 65
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
江科大51单片机定时器是指STC系列单片机中的定时器模块。STC系列单片机采用了标准的8051内核,具有多个定时器/计数器模块,用于实现定时、计数脉冲宽度调制等功能。 STC单片机定时器模块主要包括定时器0(T0)和定时器1(T1)。每个定时器都是16位的,并且可以通过不同的工作模式来实现不同的定时功能。 在STC单片机中,定时器模块的寄存器包括TCON(定时器控制寄存器)、TMOD(定时器模式寄存器)、THx(定时器高8位寄存器)和TLx(定时器低8位寄存器)等。 对于定时器0(T0),可以通过设置TMOD寄存器的相应位来选择工作模式,例如,设置TMOD的位0和位1为1,即可将T0设置为16位定时器工作模式。然后,通过设置TH0和TL0寄存器的值,可以设置定时器的初始值。 对于定时器1(T1),也可以通过设置TMOD寄存器的相应位来选择工作模式,例如,设置TMOD的位4和位5为1,即可将T1设置为16位定时器工作模式。然后,通过设置TH1和TL1寄存器的值,可以设置定时器的初始值。 在定时器开始计时之后,当定时器溢出时,将触发相应的中断(如定时器0溢出中断或定时器1溢出中断),可以在中断服务程序中进行相应的处理操作。 除了基本的定时功能外,STC单片机定时器模块还支持其他功能,如输入捕获、输出比较和脉冲宽度调制等,可以根据具体需求进行配置和使用。 以上是关于江科大51单片机定时器的简要介绍,如果有更具体的问题或需求,请提供更多详细信息。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值