} }
void disp(void) {
P0 = SEG_TAB[ sec % 10 ];//显示秒的个位 P2 = 0X01; Delay(15); P2 = 0;
P0 = SEG_TAB[ sec / 10 ];//显示秒的十位 P2 = 0X02; Delay(15); P2 = 0;
P0 = SEG_TAB[ min % 10 ];//显示分的个位 P2 = 0X04; Delay(15); P2 = 0;
P0 = SEG_TAB[ min / 10 ];//显示分的十位 P2 = 0X08; Delay(15); P2 = 0;
P0 = SEG_TAB[ hour % 10 ];//显示时的个位 P2 = 0X10; Delay(15); P2 = 0;
P0 = SEG_TAB[ hour / 10 ];//显示时的十位 P2 = 0X20; Delay(15); P2 = 0; }
void main(void) {??
while( 1 ) {
disp( ); } }??
? ? 编译烧录芯片后,观察运行现象。矣...怎么一直显示12:00:00,难道是时钟没有启动?
还是,另外的原因呢? 哦,原来是3个变量
sec,min,hour初始化后,其值一直没有改变!那我们怎么样才能让他改变数值呢?有的朋友一定会这么认为:让秒个位延时1秒,后加1,
而秒十位延时10秒后,再加1,一直加到6,分个位加1,依次类推...这样的想法是不错,但是朋友你有没有想过C语言的一般延时(除非你
把他放到中断里)极不精确!这样累计下来,一天24小时的误差,肯定很大很大,我曾经也用延时的方法写过时钟,1个小时误差8秒,那是
个什么概念!一天24小时就要24*8=192,约为3分钟,一个月就是10分钟...有没有其他的方法可以改进些呢?有!这里就要涉及到单片机中
另一个比较重要的核心部分:单片机的中断和定时器的运用!想写出比较精确(这里说的只的相对前面的做法而言比较精确而已,如果要做
更加精确的时钟,用时钟芯片比较好点,常用的有DS12887和DS1302等)的时钟程序,就一定要调用中断和定时器。还是大家先看看教材和书
吧,毕竟人家出的书,肯定比我要写的系统多了,下面我们再来简单的讲讲!
(六)
? ? 什么是中断呢?讲个比较通俗的例子:比如你正在家中看电视,突然电话响了,你的第一反应是什么?是不是先跑过去接电话!接完电话
后,继续看电视。这就是个中断的例子,中断是由电话引起了,你跑过去就是响应中断,接电话就是中断的处理!接完电话后,接续看电视, 即恢复中断,等待下个中断的到来!
? ? 但是这个好象和单片机没什么联系呀?有的朋友或许会这样疑问。是的。单片机当然不会看电视了,也不会接电话了 ! ^_^ 但是,类
比一下:比如单片机正在执行某个任务,突然要有更重要的事件,要求单片机响应,单片机就会应答响应,去执行更为重要的任务(中断处理 ),原来的任务就继续等待(现场的保护)。执行完更重要的任务后,回到中断的入口处,继续执行原来的任务(现场中断的恢复)。51系列
的单片机共有5个中断源,分别为:外中断0 、定时器T0中断、外中断1、定时器T1中断、串口中断。? ???
? ? 或许,有些朋友已经大概领会了其中的意思,有些朋友还迷迷糊糊。不过不要紧,我们继续往下看,下面我们来讲讲单片机的定时器是什 么?如何工作的?定时器,大家从字面上就可以看出其大概的意思吧?简单的说:就是起定时作用!也就是让单片机计数。定时器分为:方式
0方式1、方式2和方式3等4种工作方式。有些朋友一定会问:定时器如何启动?风扇的定时器,相信大家一定都用过吧!但是单片机的定时器,
该如何启动呢?总不该也用手一拧定时器吧! ^_^ 当然不是,我们只要给单片机一些指令,就可以启动定时器了!下面我们就定时器0,来说
说怎么启动定时器0。
TMOD = 0X01;//设置定时器0 工作方式0 TH0 = (65536 - 5000) / 256;//载入高8位初值
TL0 = (65536 - 5000) % 256;//载入低8位初值 TR0 = 1;? ?? ???//启动定时器
? ? ^_^,简单吧,这样我们就可以把定时器启动了。其中TMOD为T/C方式控制寄存器:
D7 D6 D5 D4 D3 D2 D1 D0 ? ? _? ?? ?_
? ?? ?GATE C/T? ? M1? ? M0? ? GATE C/T M1 M0
? ?? ???|_________? ? __________| |_________? ? __________| |? ?? ???T/C1? ?? ?? ? | |? ?? ???T/C0? ?? ?? ? |
? ? C/T就是counter(记数器)和timer(定时器)的选择位,若值为1,则作计数器用。为0,则为定时期用!GATE为门控位。M1和M0工作方
式的选择:若M1=0;M0=0 则为方式0:13位定时/记数器。若M1=0;M0=1则为方式1,16定时/记数器。若M1=1;M0=0则为方式2,自动装载8位
定时/记数器。若M1=1;M0=1则为方式3,只适用于T/C0,2个8位定时/记数器。
? ? 说了一大堆,感到有点困惑了吧。那我们还是来说说上面的。TMOD= 0X01;//至于为什么是0X01,大家看:我们选择的是定时器0方式0,
所以T/C1全为0,而T/C0的M1为0。M0为1,所以D0-D7为0X01;0X01表示的是16进制数,这个大家应该都知道吧!还有D0-D7表示的是2进制数。 还需要转换一下!
? ? TH0 = (65536 - 5000) / 256;//载入高8位初值。若在12M晶体下,定时5000微秒,即为5毫秒;但是如果不是在12M下,那又该怎么计算
了呢?如果是11.0592M呢?还记不记得,我们前面讲过的机器周期和时钟周期的概念? ^_^忘了,还是看看前面吧!呵呵!没事,学习嘛,忘 了再翻翻书,看看就可以了!其实上诉的5000 = 1 * C 很显然C=5000,但是如果是11.0592M那么就不是1了,应该是1.085了,那么5000 =??
1.085 * C,则C就为5000 / 1.085 = 具体多少,大家自己去算算吧?同理TL0也是一样的! 但是,细心的朋友会发现网上或者是资料上的
TH0,TL0并不是和上面一样的,而是直接TH0 = 0XEC;TL0 = 0X78 是不是和上面的一样的,别忘了单片机也是计算机的一种哦。用C的话,直 接写上计算公式就行,计算就交给单片机完成。
? ? TR0 = 1;这句就是启动定时器0,开始记数!哦,还有一点,有些朋友会问,你是65536是哪里来的呢?呵呵你可别忘了:设置定时器0??
工作方式0是16位的(2的16次方是多少,自己算算就知道了)简单吧?但是如何和中断一起使用呢?请继续看下面的讲解!
TMOD = 0X01;//设置定时器0 工作方式0 TH0 = (65536 - 5000) / 256;//载入高8位初值 TL0 = (65536 - 5000) % 256;//载入低8位初值 TR0 = 1;? ?? ???//启动定时器 ? ???
EA = 1;//开总中断
ET0 = 1;//开定时器中断。若为0则表示关闭!
? ? 这样我们,就初始化定时器T0和中断了,也就是定时器满5毫秒后,产生一次中断。产生中断后,我们怎么处理呢?嘿嘿!仔细想想? ^_^
每次中断后,我们可以让一个变量自加1,那么200次中断后,不就是1秒的时间了吗?比起上面我们说的延时来出来是不是更加精确多了呢?
那是肯定的!但是想想1秒种的时间就让单片机产生那么多次的中断,单片机会不会累着呢?恩,那么不好。如果在12M的晶体下,T0每次中
断不是可以产生最多65.336毫秒的时间吗?那么我们让他每50毫秒中断一次好了!这样我们就20次搞定一秒的时间了! ·爽·
? ? 好了,讲了那么多,现在我们来写个时间的程序吧! ^_^
#include
#define HI? ? ((65536 - 50000) / 256) #define LO? ? ((65536 - 50000) % 256)
#define _TH0_TL0_? ?? ?? ? (65536 - 50000) #define M? ? 20? ?? ?? ?? ?//(1000/25)
/**********************************************************************************************/
unsigned hou = 12, min = 0, sec = 0;
unsigned char SEG_TAB_B[ ] = {0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90}; //0-9数字
unsigned char SEG_TAB_A[ ] = {0x40,0x79,0x24,0x30,0x19,0x12,0x02,0x78,0x00,0x10};//0.-9.数字
/*********************************************************************************************/
void Delay(unsigned char a)//延时程序a*1MS {
unsigned char j;?? while(a-- != 0) {
for (j = 0; j < 125; j++); } }
/*********************************************************************************************/
void Disp(void)//数码管显示 {
P2_0 = 1;
P1 = SEG_TAB_B[ hou / 10 ];?? Delay(5); P2_0 = 0;
P2_1 = 1;
P1 = SEG_TAB_A[ hou % 10 ]; Delay(5); P2_1 = 0;
P2_2 = 1;
P1 = SEG_TAB_B[ min / 10 ];?? Delay(5);?? P2_2 = 0;
P2_3 = 1;
P1 =S EG_TAB_A[ min % 10 ];?? Delay(5); P2_3 = 0;
P2_4 = 1;
P1 = SEG_TAB_B[ sec / 10 ];?? Delay(5);?? P2_4 = 0;
P2_5 = 1;
P1 = SEG_TAB_B[ sec % 10 ];?? Delay(5);?? P2_5 = 0; }
/********************************************************************************************/
void IsrTimer0(void) interrupt 1 using 1? ? //定时50ms {
static unsigned char count = 0; //定义静态变量count
count++;
if(count == M) {
count = 0;?? sec++;??
if(sec == 60) {