51单片机定时器初值的计算
一。10MS定时器初值的计算:
1.晶振12M
12MHz除12为1MHz,也就是说一秒=1000000次机器周期。10ms=10000次机器周期。
65536-10000=55536(d8f0)
TH0=0xd8,TL0=0xf0
2.晶振11.0592M
11.0592MHz除12为921600Hz,就是一秒921600次机器周期,10ms=9216次机器周期。
65536-9216=56320(dc00)
TH0=0xdc,TL0=0x00
二。50MS定时器初值的计算:
1.晶振12M
12MHz除12为1MHz,也就是说一秒=1000000次机器周期。50ms=50000次机器周期。
65536-50000=15536(3cb0)
TH0=0x3c,TL0=0xb0
2.晶振11.0592M
11.0592MHz除12为921600Hz,就是一秒921600次机器周期,50ms=46080次机器周期。
65536-46080=19456(4c00)
TH0=0x4c,TL0=0x00
三。使用说明
以12M晶振为例:每秒钟可以执行1000000次机器周期个机器周期。而T 每次溢出
最多65536 个机器周期。我们尽量应该让溢出中断的次数最少(如50ms),这样对主程序的干扰也就最小。开发的时候可能会根据需要更换不同频率的晶振(比如c51单片机,用11.0592M的晶振,很适合产生串口
时钟,而12M晶振很方便计算定时器的时间),使用插接式比较方便。
51单片机12M和11.0592M晶振定时器初值计算2011-01-04 22:25
at89s52,晶振频率12m
其程序如下:
引用代码:#include
#include
void timer0_init()
{
TMOD=0x01;//方式1
TL0=0xb0;
TH0=0x3c;
TR0=1;
ET0=1;
}
void timer0_ISR(void) interrupt 1
{
TL0=0xb0;
.
TH0=0x3c;//50ms中断一次
single++;
if(single==20)
{
kk++;
single=0;
}
}
void main()
{
int kk=0;//计数器
int single=0;
timer0_init();
}
TL0=0xb0;
TH0=0x3c;
这两个是怎么算出来得
如果晶振不是12Mhz
是11.0592 MHz
怎么算
12M的晶振每秒可产生1M个机器周期,50ms就需要50000个机器周期,定时器在方式1工作,是16位计数器,最大值为65536,所以需设置初值15536,即
3CB0H(10进制15536转换成16进制数3CB0),所以TH0=0x3c,TL0=0xb0。(65536-50000周期=初值15536)
高位就是TH0的值,低位为TL0的值
11.0592M的晶振每秒可产生0.9216M个机器周期,50ms就需要46080个机器周期,定时器在方式1工作,是16位计数器,最大值为65536,所以需设置初值19456,即4C00H,所以TH0=0x4c,TL0=0x00。
其实很简单,不管你使用多大的晶振,使用51单片机,一般都是12分频出来,也就可以得出一个机器周期
机器周期=12/n(n指晶振频率),假设你要定时的时间为M
那么定时的初值为:
M/机器周期=初值;
TH0=(65536-初值)%256;
TL0=(65536-初值)/256;
将(65536-初值)所得的值化成16进制,其高位就是TH0的值,低位为TL0的值
例如用12M晶振做1ms定时计算如下:
机器周期=12/12*10^6=1us(微秒)
定时初值=(1*10^-3)/(1*10^-6)=1000;
所以:TH0=(65536-1000)%256;
TL0=(65536-1000)/256;
将65536-1000=64536化为16进制为:0xFC18
TH0=0xFC;
TL0=0X18;
单片机T2定时器实现1秒精确定时程序[日期:2008-07-29 ] [来源:东哥单片机学习网https://www.doczj.com/doc/761016382.html, 作者:佚名] [字体:大中小] (投递新闻)单片机T2定时器实现1秒精确定时程序
/******************************************************************************************* *
* 文件名:test.c
* 功能:使用T2定时器实现1秒精确定时并闪灯
* 1.CPU型号:AT89S52
* 2.晶振:12.000MHz
******************************************************************************************** */
#include "reg52.h" // 包含头文件
#define uint unsigned int
#define uchar unsigned char
sbit P1_7 = P1 ^ 7; // 定义位变量
/******************************************************************************************* *
* 函数名称:Timer2_Server()
* 功能:定时器2溢出中断。
* 入口参数:无
* 出口参数:无
* 注意:在本函数中设置了一个静态变量Timer2_Server_Count,静态变量的值在进入函数时是不会被
* 初始化的,而是保持上次的值。它用来计数T2定时器的溢出次数(进入本函数的次数),每
* 溢出16次,就控制一次LED11反转显示。这时的时间正好是1秒,而且是精确的1秒!只与晶振
* 的精度有关。
******************************************************************************************** */
void Timer2_Server(void) interrupt 5
{
static uint Timer2_Server_Count;
// 定义静态变量,用来计数T2定时器的溢出次数(进入本函数的次数)
TF2=0;
// T2定时器发生溢出中断时,需要用户自己清除溢出标记,而51的其他定时器是自动清除的?
Timer2_Server_Count++;
if(Timer2_Server_Count==16) // T2定时器的预装载值为0x0BDC,溢出16次就是1秒钟。
{
Timer2_Server_Count=0;
P1_7=~P1_7; // LED11反转显示。
}
}
/******************************************************************************************* *
* 函数名称:main()
* 功能:使用T2定时器实现1秒精确定时并使LED11闪灯,即LED11亮1秒,灭1秒,亮1秒,灭1秒......
* 注意:
* 1、要精确定时,必须使用定时器的自动装载方式。本实验使用T2定时器,让它工作在16bit自动
* 装载方式,这时,有另一个位置专门装着16位预装载值,当T2溢出时,预装载值立即被装入,
* 这就保证了精确定时。
* 2、T2定时器是一个16位定时器,最长的溢出时间也就几十毫秒,要定时1秒,就需要用一个变量
* 来保存溢出的次数,积累到了一定的次数后,才执行一次操作。这样就可以累加到1秒或者更
* 长的时间才做一次操作。
* 3、当T2定时器发生溢出中断时,需要用户自己清除溢出标记,而51的其他定时器是自动清除的。
* 4、T2定时器预装载值的计算:
* 设晶振为12MHz,每秒钟可以执行1000000(12000000/12)个机器周期。而T2每次溢出时最多
* 经过了65536个机器周期。我们应该尽量让T2定时器的溢出中断的次数最少,这样对主程序的干扰
* 也就最小。
* 选择每秒中断14次,每次溢出1000000/14=71428.57个机器周期,不为整数且超出6 5536个机器周期,有效。
* 选择每秒中断16次,每次溢出1000000/16=62500个机器周期,小于65536个机器周期,有效。
* 选择每秒中断20次,每次溢出1000000/20=50000个机器周期,小于65536个机器周期,有效。
* .
* .
* .
* 通过上面的计算,我们可以发现,我们可以选择的方式有很多,但是最佳的是每秒中断16次,每次
* 溢出62500个机器周期,那么赋给T2定时器的初值应为65536-62500=3036,转换成十六进制值为
* 0x0BDC。
******************************************************************************************** */
void main (void)
{
P1_7=1; // LED11灭
/* T2定时器赋预装载值,溢出16次就是1秒。 */
RCAP2H=0x0B;
RCAP2L=0xDC;
ET2=1; // 允许T2定时器中断
EA=1; // 打开总中断
TR2=1; // 启动T2定时器
while(1); // 死循环,等待T2定时器的溢出中断}