单片机笔记

定时器/计数器

定时器/计数器的结构

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.

D7 D6 D5 D4 D3 D2 D1 D0
GATE C/T’ M1 M0 GATE C/T’ M1 M0

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

M1, M0工作方式选择位
M1 M0 工作方式
0 0 方式0, 13位定时器/计数器
0 1 方式1, 16位定时器/计数器
1 0 方式2, 8位的常数自动重新装载的定时器/计数器
1 1 方式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.

D7 D6 D5 D4 D3 D2 D1 D0
TF1 TR1 TF0 TR0 IE1 IT1 IE0 IT0
TF1, TF0 - 计数溢出标志位

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

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

该位可由软件置1和清0

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

方式1(M1 = 0, M0 = 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)

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

方式2

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

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

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

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

方式3

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

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

计数器模式时, 计数脉冲来自外部输入引脚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
D7 D6 D5 D4 D3 D2 D1 D0
GATE C/T’ M1 M0 GATE C/T’ M1 M0
0 0 0 0 0 0 0 1

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闪亮* 编写环境: Keil5* 硬件仿真: Proteus 8 Professional* 日期: 2022-11-05*/#include <reg51.h>char i = 100; // 用于软件计数int main(void)
{TMOD = 0x01; // 定时器T0为方式1TH0 = 0xee; // 设置定时器初值TL0 = 0x00;P1 = 0x00; // P1口8个LED点亮EA = 1; // 总中断开ET0 = 1; // T0中断开TR0 = 1; // 启动T0while(1) // 循环等待{;}return 0;
}void timer0() interrupt 1 // T0中断程序
{TH0 = 0xee; //重新赋初值TL0 = 0x00;// 软件计数, 100次溢出中断后, LED状态进行转换  i--; // 循环次数减1if (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;   // 启动定时器T1while (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);   // 延时500msP1 = 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口为Pulseint main(void)
{TMOD = 0x01; // 定时器T0为方式1TH0 = 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; // 启动T0while(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口为Pulseunsigned char count = 0;int main(void)
{TMOD = 0x01; // 定时器T0为方式1TH0 = (65536 - 2000) / 256; // 设置定时器初值TL0 = (65536 - 2000) % 256;Pluse = 0; // P1.1输出低电平EA = 1; // 总中断开ET0 = 1; // T0中断开TR0 = 1; // 启动T0while(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二者数值相同, 不存在将十六进制数分位的问题


源码
/* * 功能实现:  使用T0, 采用方式2定时中断, 在P1.0引脚上输出周期为400us, 占空比为25%的矩形脉冲 * 编写环境: Neovim + Keil5* 硬件仿真: Proteus 8 Professional* 日期: 2022-11-15*/#include <reg51.h>sbit Pluse = P1^0; // 定义P1.0口为Pulseunsigned char count = 0;int main(void)
{TMOD = 0x01; // 定时器T0为方式1TH0 = 256 - 100; // 设置定时器初值TL0 = 256 - 100; Pluse = 0; // P1.1输出低电平EA = 1; // 总中断开ET0 = 1; // T0中断开TR0 = 1; // 启动T0while(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;
}

51单片机笔记:定时器/计数器相关推荐

  1. 51单片机之定时器\计数器的工作原理

    51单片机之定时器\计数器的工作原理 定时器/计数器的结构: 何时处于计数器方式?何时处于定时器方式? TCON和TMOD特殊功能寄存器: 定时器/计数器工作方式: 方式0:13位的定时器/计数器 方 ...

  2. 【51单片机】定时器/计数器的工作原理和结构(一)

    [51单片机]定时器/计数器的工作原理和结构 回看下单片机中断系统内部结构: 分析内部中断触发: 先看图 TCON:和外部中断相比少了IT位设置触发方式 :1跳沿触发 0电平触发:此处我的理解是:就好 ...

  3. 六、51单片机之定时器/计数器_理论

    1.什么是定时器.计数器 定时器就是单片机设定一个时间间隔,时间间隔到后通知单片机.例如设置100ms的定时器,100ms后定时器通知单片机时间到了. (1)定时器是单片机的一种内部外设.(以前的单片 ...

  4. 51单片机之定时器/计数器应用实例(方式0、1、2、3)

    硬件:STC89C52RC 开发工具:Keil uVision4 对于刚接触单片机的同学来说可能会对定时器/计数器的应用很蒙圈,特别是初值的计算和各种定时方式的选择.下面希望能给你带来一个清晰的思路. ...

  5. 51单片机内部定时器/计数器实验

    51单片机内部定时器/计数器实验 一.实验内容 使用MCS-51内部定时/计数器,定时1秒钟,CPU运用定时中断方式,实现每1秒钟输出状态发生一次反转,即发光管每隔1秒钟亮一次(P1.7接LED). ...

  6. 51单片机计算定时器初值

    51单片机计算定时器初值 前言 理论分析 工作方式寄存器 TMOD GATE 门控位 C/T' 计数器模式和定时器模式选择位 M1 M0 工作方式选择位 定时器/计数器控制寄存器 TCON TCON补 ...

  7. 51单片机的定时/计数器

    51单片机的定时/计数器 实现LED灯闪烁,代码如下: #include <reg51.h> //51单片机头文件 sbit LED = P0^0; //定义特殊寄存器p0.0;注意:sb ...

  8. 51单片机学习--定时器--中断--串口通信

    51单片机学习–定时器–中断–串口通信 定时器–中断–串口通信 中断分类 定时器中断 外部中断 串口中断 基本概念 对于单片机来讲, 中断是指 CPU 在处理某一事件 A 时, 发生了另一事件 B, ...

  9. (六)51单片机基础——定时器

    定时器介绍:51单片机的定时器属于单片机的内部资源,其电路的连接和运转均在单片机内部完成. 定时器主要作用: 用于计时系统,可实现软件计时,或者使程序每隔一固定时间完成一项操作. 替代长时间的Dela ...

最新文章

  1. 讯飞智慧餐厅(全国赛区)比赛通知
  2. mxnet基础到提高(27)-Dense
  3. word手写字体以假乱真_学会Word上下标,搞定公式输入
  4. python人工智能——机器学习——分类算法-k近邻算法
  5. P1137-旅行计划【拓扑排序,DAGdp】
  6. maven仓库理解、下载及设置
  7. 如何训练自己的编程思路
  8. MYSQL-实现ORACLE- row_number() over(partition by ) 分组排序功能
  9. 算法竞赛入门经典(刘汝佳版)例题与解答
  10. 清空H3C交换机CF存储回收站空间
  11. 【入门】倒序输出一个四位整数
  12. compose安装单个扩展_将自动化测试扩展到单个项目之外
  13. Android最火的框架系列(五)GreenDao
  14. 如何查看、复制caj、PDF文件文本内容
  15. 元宇宙创造的新世界变成了统一的个体
  16. apk分享: Android应用更换皮肤功能的实现思路教程。
  17. windows安装Cygwin以及swoole,方便本地开发仅支持linux环境的easyswoole。
  18. halo 开源项目源码学习
  19. l310加完墨水后需要怎样设置_爱普生打印机加了墨后为何还打印不出来?
  20. 毕业设计《宠物医院挂号系统》

热门文章

  1. Exce丨服务器2O19系统己满,《电子表格Exce20.docx
  2. 路由守卫的几种方法、全局守卫、路由独享守卫、组件内部守卫
  3. Java-MVC三层架构
  4. camstar portal使用webapi,vue elementui前后端分离开发
  5. 一个C语言程序是由( )组成?
  6. 013、full-page-writes(全页写)
  7. C语言--压缩字符串
  8. Java:最新eclipse java安装教程2022.11.20
  9. mac系统下,怎么装win7系统?
  10. 数据质量管理_第四篇 对数变换