51单片机定时器和中断的介绍

发布者:ZenMaster123最新更新时间:2023-10-12 来源: elecfans关键字:51单片机  定时器  中断 手机看文章 扫描二维码
随时随地手机看文章

最近在学习51单片机,学到了定时器这块,由于自己的基础不太扎实,在这方面花了很多时间,这里通过对定时器和中断的介绍,用简易时钟这个例子来对学习的内容进行加深巩固,把自己的经验分享给大家,希望对大家能够有帮助。


一、定时器的功能以及定时器的结构


定时器的功能

其实就是单片机的内部,通过系统时钟的每一个机器周期产生一个记数脉冲,即每一个机器周期计数器加一。

比如,这里我的实验板的晶振是12MHZ,1MHZ信号每个脉冲的持续时间为1us,如果定时器T0对1MHZ的信号进行计数,从0~65536us,当达到最大的65536us的时候,定时器计数达到最大值,会溢出,于是产生中断信号,向中断系统申请中断,中断系统接受中断请求,执行中断子程序。

定时器的结构

定时器的结构如下图所示,主要包括

  • 两个定时器/计数器。T0和T1,每个定时器/计数器都是由两个8位的计数器所构成的16位计数器。

  • TCON 寄存器。TCON为控制寄存器,用来控制两个定时器/计数器的启动和停止。

  • TMOD寄存器。TMOD为工作方式控制寄存器,用来设置定时器/计数器的工作方式。

图片

二、定时器的控制

工作模式寄存器TMOD

TMOD为工作方式控制寄存器,用来设置定时器/计数器的工作方式。如下图所示。

通过配置TMOD寄存器来对定时器T0和T1的工作模式进行控制。

注意这里TMOD的地址为89H,不可位寻址。

TMOD的高四位用于T1,低四位用于T0。

其中主要各位的功能:

  • C/-T,定时器/计数器的选择控制位。置0,为定时器模式,置1,为计数器模式。

  • M1和M0,模式选择控制位,通过对两位进行赋值,可以选择定时器的4种模式。00,模式0,13位计数器。01,模式1,16位计数器。10,模式2,自动重装8位计数器,11,模式3。

  • GATE,置1后,就可由TR0或TR1单独控制定时器。

图片

控制寄存器TCON

TCON寄存器,地址为88H,可以字节寻址,也可位寻址。寄存器各位如下图所示。

其中各位的功能:

  • TF1(TF0)。定时器T1(T0)溢出标志位。当T1(T0)溢出时,硬件自动使TF1(TF0)置1,并且向cpu申请中断。当Cpu响应中断,进入中断服务子程序后,TF1(TF0)由硬件自动清0,当然也可以用软件写代码清0。

  • TR0(TR1)。定时器T1(T0)运行控制位,置1,定时器T1(T0)就开始运行,计数。

  • 后面几个是外部中断控制位。

图片

写代码来初始化定时器

定时器的配置主要是通过对两个寄存器TMOD和TCON进行配置,这里我通过配置定时器0,模式1引发中断,配置其他的定时器或者是不同的模式都是大同小异。看看模式1的结构。

模式1的结构

图片

好,我们首先来配置寄存器TMOD,根据图来配置。

图片

只需要配置定时器0,那么高四位就不管了,置0,而我们在控制定时器0的低四位中配置为0001。

GATE=0; //直接由TR0控制定时器0的开启

C/-T=0; //选择定时器模式

M1=0; //选择模式1

M2=1;

继续配置寄存器TCON。

图片

只需要配置定时器0相关的部分就可以了,再一个,TCON寄存器是可位寻址的,所以只需要单独对其中的某一位进行置值就可以了。

所以:

TF0=0; //定时器0溢出控制标志,当计数到溢出65536us时,就会置1。

TR0=1; //定时器0启动,开启计时。

配置中断

当计数到溢出后,就会向cpu发出中断请求,申请中断,进入中断子程序。然后出来,TF0由1->0,然后循环循环。

所以:

ET0=1; //中断的配置

EA=1;

PT0=0;

三、定时器引发中断

简易时钟

使用定时器,采用LCD1602,实现简易时钟,秒,分,时。

下面是源代码:


主程序main.c


#include < REGX52.H >

#include "Delay.h"

#include "Timer0.h"

#include "LCD1602.h"

unsigned char sec=55,min=59,hour=23;

void main()

{

  LCD_Init();  //LCD初始化

  LCD_ShowString(1,1,"COLCK:");

  Timer0Init();  //定时器0初始化

  while(1)

  {

    LCD_ShowNum(2,1,hour,2); 

    LCD_ShowString(2,3,":");

    LCD_ShowNum(2,4,min,2);

    LCD_ShowString(2,6,":");

    LCD_ShowNum(2,7,sec,2);

  }

}





void TimerRoutine() interrupt 1

{

  static unsigned int T0Count;

  //当触发中断后,每次中断结束后,初始值还是为64535 即1ms

  TL0 = 0x66;    //设置定时初值

  TH0 = 0xFC;    //设置定时初值

  T0Count++;

  if(T0Count >=1000)  //一次是1ms,*1000就是一秒    

  {

    T0Count=0;

    sec++;

    if(sec >=60)

    {

      sec=0;

      min++;

      if(min >=60)

      {

        min=0;

        hour++;

        if(hour >=24)

        {

          hour=0;

          sec=0;

          min=0;

        }

      }

    }

  }

}

延时函数Delay.c


//延时

void Delay(unsigned char xms)    //@11.0592MHz

{

  unsigned char i, j;



  while(xms--)

  {

    //_nop_();

  i = 2;

  j = 199;

  do

  {

    while (--j);

  } while (--i);

  }

}

控制LCD162模块LCD1602.c


虽然还不怎么懂这个模块,但是可以直接用,模块都写好了的。后面应该会弄懂各个函数功能如何实现。


#include < REGX52.H >



//引脚配置:

sbit LCD_RS=P2^6;

sbit LCD_RW=P2^5;

sbit LCD_EN=P2^7;

#define LCD_DataPort P0



//函数定义:

/**

  * @brief  LCD1602延时函数,12MHz调用可延时1ms

  * @param  无

  * @retval 无

  */

void LCD_Delay()

{

  unsigned char i, j;



  i = 2;

  j = 239;

  do

  {

    while (--j);

  } while (--i);

}



/**

  * @brief  LCD1602写命令

  * @param  Command 要写入的命令

  * @retval 无

  */

void LCD_WriteCommand(unsigned char Command)

{

  LCD_RS=0;

  LCD_RW=0;

  LCD_DataPort=Command;

  LCD_EN=1;

  LCD_Delay();

  LCD_EN=0;

  LCD_Delay();

}



/**

  * @brief  LCD1602写数据

  * @param  Data 要写入的数据

  * @retval 无

  */

void LCD_WriteData(unsigned char Data)

{

  LCD_RS=1;

  LCD_RW=0;

  LCD_DataPort=Data;

  LCD_EN=1;

  LCD_Delay();

  LCD_EN=0;

  LCD_Delay();

}



/**

  * @brief  LCD1602设置光标位置

  * @param  Line 行位置,范围:1~2

  * @param  Column 列位置,范围:1~16

  * @retval 无

  */

void LCD_SetCursor(unsigned char Line,unsigned char Column)

{

  if(Line==1)

  {

    LCD_WriteCommand(0x80|(Column-1));

  }

  else if(Line==2)

  {

    LCD_WriteCommand(0x80|(Column-1+0x40));

  }

}



/**

  * @brief  LCD1602初始化函数

  * @param  无

  * @retval 无

  */

void LCD_Init()

{

  LCD_WriteCommand(0x38);//八位数据接口,两行显示,5*7点阵

  LCD_WriteCommand(0x0c);//显示开,光标关,闪烁关

  LCD_WriteCommand(0x06);//数据读写操作后,光标自动加一,画面不动

  LCD_WriteCommand(0x01);//光标复位,清屏

}



/**

  * @brief  在LCD1602指定位置上显示一个字符

  * @param  Line 行位置,范围:1~2

  * @param  Column 列位置,范围:1~16

  * @param  Char 要显示的字符

  * @retval 无

  */

void LCD_ShowChar(unsigned char Line,unsigned char Column,char Char)

{

  LCD_SetCursor(Line,Column);

  LCD_WriteData(Char);

}



/**

  * @brief  在LCD1602指定位置开始显示所给字符串

  * @param  Line 起始行位置,范围:1~2

  * @param  Column 起始列位置,范围:1~16

  * @param  String 要显示的字符串

  * @retval 无

  */

void LCD_ShowString(unsigned char Line,unsigned char Column,char *String)

{

  unsigned char i;

  LCD_SetCursor(Line,Column);

  for(i=0;String[i]!='�';i++)

  {

    LCD_WriteData(String[i]);

  }

}



/**

  * @brief  返回值=X的Y次方

  */

int LCD_Pow(int X,int Y)

{

  unsigned char i;

  int Result=1;

  for(i=0;i< Y;i++)

  {

    Result*=X;

  }

  return Result;

}



/**

  * @brief  在LCD1602指定位置开始显示所给数字

  * @param  Line 起始行位置,范围:1~2

  * @param  Column 起始列位置,范围:1~16

  * @param  Number 要显示的数字,范围:0~65535

  * @param  Length 要显示数字的长度,范围:1~5

  * @retval 无

  */

void LCD_ShowNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length)

{

  unsigned char i;

  LCD_SetCursor(Line,Column);

  for(i=Length;i >0;i--)

  {

    LCD_WriteData(Number/LCD_Pow(10,i-1)%10+'0');

  }

}



/**

  * @brief  在LCD1602指定位置开始以有符号十进制显示所给数字

  * @param  Line 起始行位置,范围:1~2

  * @param  Column 起始列位置,范围:1~16

  * @param  Number 要显示的数字,范围:-32768~32767

  * @param  Length 要显示数字的长度,范围:1~5

  * @retval 无

  */

void LCD_ShowSignedNum(unsigned char Line,unsigned char Column,int Number,unsigned char Length)

{

  unsigned char i;

  unsigned int Number1;

  LCD_SetCursor(Line,Column);

  if(Number >=0)

  {

    LCD_WriteData('+');

    Number1=Number;

  }

  else

  {

    LCD_WriteData('-');

    Number1=-Number;

  }

  for(i=Length;i >0;i--)

  {

    LCD_WriteData(Number1/LCD_Pow(10,i-1)%10+'0');

  }

}



/**

  * @brief  在LCD1602指定位置开始以十六进制显示所给数字

  * @param  Line 起始行位置,范围:1~2

  * @param  Column 起始列位置,范围:1~16

  * @param  Number 要显示的数字,范围:0~0xFFFF

  * @param  Length 要显示数字的长度,范围:1~4

  * @retval 无

  */

void LCD_ShowHexNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length)

{

  unsigned char i,SingleNumber;

  LCD_SetCursor(Line,Column);

  for(i=Length;i >0;i--)

  {

    SingleNumber=Number/LCD_Pow(16,i-1)%16;

    if(SingleNumber< 10)

    {

      LCD_WriteData(SingleNumber+'0');

    }

    else

    {

      LCD_WriteData(SingleNumber-10+'A');

    }

  }

}



/**

  * @brief  在LCD1602指定位置开始以二进制显示所给数字

  * @param  Line 起始行位置,范围:1~2

  * @param  Column 起始列位置,范围:1~16

  * @param  Number 要显示的数字,范围:0~1111 1111 1111 1111

  * @param  Length 要显示数字的长度,范围:1~16

  * @retval 无

  */

void LCD_ShowBinNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length)

{

  unsigned char i;

  LCD_SetCursor(Line,Column);

  for(i=Length;i >0;i--)

  {

    LCD_WriteData(Number/LCD_Pow(2,i-1)%2+'0');

  }

}

定时器0模块Timer0.c


主要是对定时器进行配置,看了视频第一遍没有听懂,之后回头再去看这个定时器,发现其实也没有很难,只是自己的畏难情绪罢了。


别放弃,你可以弄明白的,只是心理在作祟。


#include < REGX52.H >



/**

  * @brief 定时器0初始化

  * @param  

  * @retval 

  */

void Timer0Init()

{

  TMOD&=0xF0;  //高四位不变

  TMOD|=0x01;  //设置定时器模式1 以及设置为定时方式 0

  

  TL0 = 0x66;    //设置定时初值

  TH0 = 0xFC;    //设置定时初值

  

  TF0=0;  //定时器0溢出标志位

  TR0=1;  //定时器0运行控制位

  

  TF0=1;  //设置外部中断

  ET0=1;

  EA=1;

  PT0=0;

}

实现效果

如下图。

图片

自己卡着时间哈哈,还是慢了一秒。

图片

总结

定时器的配置主要是通过配置,两个寄存器TMOD和TCON。

在配置时,只要明确要配置的要求,一步一步来,也不难的喔!

明确要配置的是定时器还是计数器,是模式1还是模式几。TCON寄存器TR0(TR1)置1,定时器启动开始运行,和TF0(TF1),一般都是置0。

然后如果要配置中断的话,根据外部中断查看手册来进行配置,一般也只需要配置几个就可以了。

再者,一步一步好好学,没有什么难的。


关键字:51单片机  定时器  中断 引用地址:51单片机定时器和中断的介绍

上一篇:3步教你如何用51单片机制作秒表
下一篇:51单片机对智能防火防盗报警系统的设计

推荐阅读最新更新时间:2024-11-17 00:28

STM8S---外部中断应用之长按键识别
STM8常用中断指令 开总中断 _asm(“rim”); 禁止中断 _asm(“sim”); 进入停机模式 _asm(“halt”); 中断返回 _asm(“iret”); 等待中断 _asm(“wfi”); 软件中断 _asm(“trap”); STM8S常用中断映射 如使用中断函数时,可以通过在上图中查找相对应的中断向量号,而中断函数的名字可以自定义 /* BASIC INTERRUPT VECTOR TABLE FOR STM8 devices * Copyright (c) 2007 STMicroelectronics */ typedef vo
[单片机]
采用LMV1090放大器和AT89C51单片机改进型噪声抑制话音采集方案
机载超短波电台在战斗机中承担着与地面塔台及友机之间的通话任务,其通话质量的高低直接影响飞行员的作战与训练质量。目前在飞机上大量使用的还是老式的模拟话音电台,此种电台缺少语音增强装置,对于战斗机座舱内强烈的噪声缺少针对性的抑制措施。随着新阶段飞行任务的增大,低清晰度的通话质量势必会对飞行任务的完成及飞行员的身心健康造成较大的影响。本文拟通过新型的模拟语音处理方法,重新设计话音采集及控制部分,提高采集语音的信噪比,改善机载超短波电台的通信质量。 1、 机载电台话音采集装置 某型飞机超短波电台由控制器、收发机、功率附加器、射频转换器及通信天线构成。该电台的话音采集装置主要包括麦克风和控制器,其中控制器通过座椅连接器与飞行帽上的耳机及
[单片机]
采用LMV1090放大器和AT89C<font color='red'>51单片机</font>改进型噪声抑制话音采集方案
基于Intel8274的机载同步通讯系统设计
  异步串行通讯,诸如RS232、RS485等以简单实用而众所周知。同步串行通讯以异步不可比拟的高速度而应用在许多速度要求较高的通讯网络。同步通信无需开始位和停止位,直接由同步时钟对数据采样,数据传输率等同于时钟速率。其显著特点就是两端必须共享同一时钟,发送方必须将时钟和数据同时发送,接收方才能正确采样。同步模式分为字节同步和位同步。国际标准化组织的HDLC(高级数据链路控制)协议是应用最普遍的同步协议,帧结构如表1所示,其中标志字节值为7EH。 Intel8274简介   Intel8274是实现HDLC同步通信协议的最佳接口芯片,图1是其管脚定义,有以下显著特点: ·有异步、字节同步和位同步工作模式; ·2个独立的、全双工的
[网络通信]
STM32是如何进入中断服务函数xxx_IRQHandler的
今天在看stm32的中断,一时间不理解stm32主函数是如何进入中断函数的,按C编程的理解,会有个特定的入口之类的,但是看demo过程中没有发现入口。 以串口中断服务函数void USART1_IRQHandler(void) 为例,首先用到串口中断,需要先设定串口中断初始化以及串口初始化,另外void USART1_IRQHandler(void) 中断服务函数也应该写好。 发现在stm32的启动文件startup_stm32f10x_md.s中写到 DCD USART1_IRQHandler 其中DCD是一条数据定义伪指令,用于分配一片连续的字存储单元并用指定的数据初始化。 库里定义 #define USART1 ((
[单片机]
51单片机---存储器
一、AT89C51单片机的整体存储结构 二、AT89C51单片机的程序存储器 三、AT89C51单片机的数据存储器 简单结构: 详细结构:
[单片机]
<font color='red'>51单片机</font>---存储器
51单片机驱动8位数码管电路图+程序
电路图: JP10排线连接J12 J21跳线跳12处 测试程序 #include reg52.h typedef unsigned char BYTE; typedef unsigned int WORD; sbit LS138A = P2^2; //定义138译码器的输入A脚由P2.2控制 sbit LS138B = P2^3; //定义138译码器的输入脚B由P2.3控制 sbit LS138C = P2^4; //定义138译码器的输入脚C由P2.4控制 //此表为 LED 的字模, 共阴数码管 0-9 - BYTE code Disp_Tab = {0x3f,0x06,0
[单片机]
<font color='red'>51单片机</font>驱动8位数码管电路图+程序
TimerInterrupt.h头文件下载-AVR的通用定时/计数器中断控制
/***************************************************************************** 文件名称:PORT.H 文件标识:_PORT_H_ 摘 要:AVR单片机的通用定时/计数器中断控制头文件 当前版本:V1.0 *****************************************************************************/ #ifndef _TIMERINTERRUPT_H_ #define _TIMERINTERRUPT_H_ #define OC2 (1 7)
[单片机]
基于ARM处理器的MVB 2类设备研究
1 引 言 列车需要传输大量的设备控制和旅客服务信息,随着这些信息的数量和种类不断地增长,迫切需要一种大容量,高速度的信息传输系统。为此,国际电工委员会(IEC)制定了一项用于规范车载设备数据通信的标准——IEC61375(列车通信网标准),即TCN标准,该标准于1999年6月成为国际标准。目前国际上主要的TCN产品供应商是德国西门子和瑞士Duagon公司,国内的株洲电力机车研究所和大连北车集团电力牵引研究所等单位进行了大量的TCN相关研究工作并取得了丰硕的科研成果。 TCN标准推荐在机车上层使用绞线式列车总线WTB,在下层使用多功能车辆总线MVB。MVB总线和机车中的各种电气设备相连,这些设备按性能可以分为5类,其中二类设备
[工业控制]
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved