可位寻址和不可位寻址区别,立即寻址与直接寻址有何本质区别

首页 > 实用技巧 > 作者:YD1662023-11-11 02:56:41

基本数字逻辑门电路

不管是数字电路,还是C语言,我们都会经常遇到逻辑运算和逻辑电路。

首先,在“逻辑”这个概念范畴内,存在真和假这两个逻辑值,而将其对应到数字电路或C语言中,就变成了“非0值”和“0值”这两个值,即逻辑上的“假”就是数字电路或C语言中的“0”这个值,而逻辑“真”就是其它一切“非0值”。

然后,来具体分析一下几个主要的逻辑运算符。假定有2个字节变量:A和B,二者进行某种逻辑运算后的结果为F。以下逻辑运算符都是按照变量整体值进行运算的,通常就叫做逻辑运算符:

  • &&:逻辑与,F = A && B,当A、B的值都为真(即非0值,下同)时,其运算结果F为真(具体数值为1,下同);当A、B值任意一个为假(即0,下同)时,结果F为假(具体数值为0,下同)。
  • ||:逻辑或,F = A || B,当A、B值任意一个为真时,其运算结果F为真;当A、B值都为假时,结果F为假。
  • ! :逻辑非,F = !A,当A值为假时,其运算结果F为真;当A值为真时,结果F为假。
  • 以下逻辑运算符都是按照变量内的每一个位来进行运算的,通常就叫做位运算符:

    要看资料或芯片手册的时候,会经常遇到一些电路符号,表1就是数字电路中的常用符号,知道这些符号有利于我们理解器件的逻辑结构,尤其重点认识以下表中的“国外流行图形符号”。在这里我们先简单看一下,后边遇到了知道到这里查阅就可以了。

    定时器

    定时器是单片机的重点中的重点,但不是难点,大家一定要完全理解并且熟练掌握定时器的应用。

    这两个概念了解即可,下边就来重头戏,定时器和计数器。定时器和计数器是单片机内部的同一个模块,通过配置SFR(特殊功能寄存器)可以实现两种不同的功能,大多数情况下是使用定时器功能,计数器功能大家自己了解下即可。

    顾名思义,定时器就是用来进行定时的。定时器内部有一个寄存器,让它开始计数后,这个寄存器的值每经过一个机器周期就会加1一次,因此,可以把机器周期理解为定时器的计数周期。秒表每经过一秒,数字加1,而这个定时器就是每过一个机器周期的时间,也就是12/11059200秒,数字加1。

    还有一个特别注意的地方,就是秒表是加到60后,秒就自动变成0了,这种情况在单片机和计算机里称之为溢出。那定时器加到多少才会溢出呢?定时器有几种模式,假如是16位的定时器,也就是2个字节,最大值就是65535,那么加到65535后,再加1就算溢出,如果有其他位数的话,道理是一样的,对于51单片机来说,溢出后,这个值会直接变成0。从某一个初值,经过计算确定的时间后溢出,这个过程就是其定时的含义。

    定时器的寄存器描述

    标准的51里边只有定时器0和定时器1这两个定时器,现在很多单片机也有多个定时器的,在这里先讲定时器0和1。前边提到过,对于单片机的每一个功能模块,都是由他的SFR,也就是特殊功能寄存器来控制。而和定时器有关的特殊功能寄存器,有以下几个,大家不需要去记忆这些寄存器的名字和作用,你只要大概知道就行,用的时候,随时可以查手册,找到每个寄存器的名字和每个寄存器所起到的作用。

    可位寻址和不可位寻址区别,立即寻址与直接寻址有何本质区别(1)

    表2 定时值存储寄存器

    可位寻址和不可位寻址区别,立即寻址与直接寻址有何本质区别(2)

    表3 TCON--定时器/计数器控制寄存器的位分配(地址:88H)

    表3中有TF1、TR1、TF0、TR0这4位需要理解清楚。两位定时器1的,两位定时器0的,只解释定时器1的,定时器0的同理。先看TR1,当我们程序中写TR1 = 1以后,定时器值就会每经过一个机器周期加1,当程序中写TR1 = 0以后,定时器值就会保持不变化。TF1,这个是一个标志位,它的作用是告诉定时器溢出了。
    比如定时器设置成16位的定时器,那么每经过一个机器周期,TL1加1一次,当TL1加到255后,再加1,TL1变成0,TH1会加1一次,如此一直加到TH1和TL1都是255(即TH1和TL1组成的16位整型数为65535)以后,再加1一次,那么就会溢出,TH1和TL1同时都变为0,只要一溢出,TF1马上自动变成1,告诉定时器溢出了,仅仅是提供给一个信号,让知道定时器溢出了,它不会对定时器是否继续运行产生任何影响。

    可位寻址和不可位寻址区别,立即寻址与直接寻址有何本质区别(3)

    表4 TCON--定时器/计数器控制寄存器的位描述

    可位寻址和不可位寻址区别,立即寻址与直接寻址有何本质区别(4)

    表5 TMOD--定时器方式控制寄存器的位分配(地址:89H)

    TCON是“可位寻址”,TMOD是“不可位寻址”。这个地方的意思就是比如TCON有一位TR1,我们可以在程序中直接进行TR1 = 1,这样操作。但是(T1)M1 = 1,这样的操作就是错误的,操作就必须一次操作一个字节,就是必须一次性对TMOD所有位操作,不能对其中某一位单独进行操作。

    可位寻址和不可位寻址区别,立即寻址与直接寻址有何本质区别(5)

    表7 TMOD--定时器方式控制寄存器M1/M0工作模式

    以上这4种模式的配置,其中模式0是为了兼容老的8048单片机而设的,现在的51几乎不会用到这种模式,而模式3根据应用经验,他的功能模式2完全可以取代,所以基本上也是不用,重点就学习模式1和模式2。

    理解定时器原理

    为了加深大家理解这个定时器原理,来看一下模式1的电路示意图1。

    可位寻址和不可位寻址区别,立即寻址与直接寻址有何本质区别(6)

    图1 定时器/计数器模式1示意图

    分析一下这个示意图,OSC框表示时钟频率,因为1个机器周期等于12个时钟周期,所以那个d就等于12。下边GATA右边的那个门是一个非门电路,再右侧是一个或门,再往右是一个与门电路。

    图上可以看出来,下边部分电路是控制了上边部分,那我们先来看下边是如何控制的,我们以定时器0为例。

    定时器程序应用

    了解了定时器相关的寄存器,那么我们下面就来做一个定时器的程序,巩固一下我们学到的内容。我们这节课的程序先使用定时器0,在使用定时器的时候,需要以下几个步骤:

    1. 设置特殊功能寄存器TMOD,配置好工作模式;
    2. 设置计数寄存器TH0和TL0的初值;
    3. 设置TCON,通过打开TR0位来让定时器开始计数。
    4. 判断TCON寄存器的TF0位,监测定时器溢出情况。

    写程序之前,要先来学会计算如何用定时器定时时间。以晶振是11.0592M为例讲解,时钟周期就是1/11059200,机器周期就是12/11059200,假如要定时20ms就是0.02秒,要经过x个机器周期得到0.02秒,算一下x*12/11059200 = 0.02,得到x = 18432。

    那么现在16位的定时器溢出值是65536,可以这样,先给TH0和TL0一个初值,让他们经过18432个机器周期后刚好溢出,溢出后可以通过检测TF0位得知,就刚好是0.02秒。这个初值y = 65536 - 18432 = 47104,转成16进制就是0xB800,那么就是TH0 = 0xB8,TL0 = 0x00。

    那0.02秒已经定时出来了,细心的同学会发现,如果初值直接给一个0x0000,一直到65536溢出,定时器定时值最大也就是71ms左右,那么想定时更长时间怎么办呢?用你小学学过的逻辑,倍数关系就可以解决此问题。

    本程序实现的结果是小灯点亮持续一秒,熄灭持续一秒,也就是以0.5HZ的频率进行闪烁。那好了,我们下面就用程序来实现以下这个功能。

    #include //包含寄存器的库文件

    sbit LED = P0^0;

    sbit ADDR0 = P1^0;

    sbit ADDR1 = P1^1;

    sbit ADDR2 = P1^2;

    sbit ADDR3 = P1^3;

    sbit ENLED = P1^4;

    void main()

    {

    unsigned char counter = 0;

    ENLED = 0; ADDR0 = 0; ADDR1 = 1;

    ADDR2 = 1; ADDR3 = 1; LED = 1; //74HC138和LED灯初始化部分

    TMOD = 0x01; //设置定时器0为模式1

    TH0 = 0xB8;

    TL0 = 0x00; //定时值初值

    TR0 = 1; //打开定时器0

    while(1)

    {

    if(1 == TF0) //判断定时器0是否溢出

    {

    TF0 = 0;

    TH0 = 0xB8; //一旦溢出后,重新赋值

    TL0 = 0x00;

    counter ;

    if(50 == counter) //判断定时器0溢出是否达到50次

    {

    counter = 0; //counter清0,重新计数

    LED = !LED; //LED取反操作,0-->1,1-->0

    }

    }

    }

    }

    程序都有注释,不难理解,这里要解释一个地方,就是两次if判断,细心的同学会发现,if(1 == TF0)这句,1写前边,因为如果写if(TF0 == 1),作为新手来说,不小心丢掉一个’=’号后,写成if(TF0 = 1),这样实际上在语法上是可以通过的,用的Keil4还会出一个警告说明一下,Keil以前的版本以及一些其他软件,可能根本不会出任何错误或者警告提示,但是这样产生的Hex文件下载到单片机里边,程序就错了,大家可以改改试试看。

    栏目热文

    文档排行

    本站推荐

    Copyright © 2018 - 2021 www.yd166.com., All Rights Reserved.