Q2:#define OSC_FREQ 22118400L这句宏命令里的“L”是什么意思?
Q4:为什么好多变量都是char类型?它不是字符类型吗?怎么可以用来计数?
Q4.1:51单片机中的char,int,long,float,double各占多少个字节,取值范围多大?
Q6:void timer() interrupt 1 using 2是什么意思?
Q8:经常看到TH0与TL0,例如TH0 = 0xD8;TL0 = 0xEF;这起什么作用?
Q2:P3口的8个引脚有哪些复用功能(第二功能),默认开启吗?
Q3:我的程序编译后生成的HEX文件超过了8k,烧进单片机不会有问题吗?
问题解决2:Keil生成超过8K的HEX文件会报错,提示Target not created
程序示例1:蜂鸣器滴两次、进入中断服务——数码管显示8,延迟1秒后熄灭。
Q5:为了理解某段程序的作用,我需要详细了解一下计时/计数系统。
程序示例3:蜂鸣器持续滴滴,利用定时器完成Nms的延时,进入中断服务——数码管显示8,延迟1秒后熄灭。
Q7:单片机的有好多特殊寄存器,我需要总结一下他们的名称及用途
最近做比赛,需要写程序做一个智能小车。C语言的基础和编程的能力我是有的,但是我对单片机等硬件不是很了解,特意进行了一番学习。估计以后也用不了多少,特此写一篇笔记,方便后人参考学习。
我不喜欢翻着教材或视频一节一节地学习,我的学习方式是问题启发式学习:直接切入正题,遇到不会的问题就找度娘,学会之后再次进入正题,遇到问题再查阅资料,循环往复,直到走通为止~整个学习下来,虽然可能会有些漏洞,但是已经基本进入状态了。
由于我是业余的,所以难免会有错解或不妥之处,还请读者能以挑剔的眼光为我指出。
一、C语言相关
Q1:sbit与sfr代表是什么?有什么作用?
A1:sfr用来声明特殊功能的寄存器,sbit用来声明特殊功能位。
位,取值范围为0 ~ 255 = 2^8-1。对于I/O端口来说,刚好每一位对应一个引脚),例如 sfr P0 = 0x80; 这一句定义了P0端口与地址0x90对应。特殊功能的寄存器一般在开发工具(Keil)中自带的头文件,例如reg52.H中声明好了,只需要在程序中引入该头文件就好了;
w 用法:sfr 变量名 = 地址值;
w 需要注意的一点是,例如P0对应的是一个”8“字型的数码管,若要显示3,则可对P0口赋值:P0 = 0x0D,若要将其熄灭,只需对其赋值:P0=0xFF。进制的数进制,后两位。刚好FF代表十进制的255)。
w sfr16也是用来声明特殊功能寄存器,所不同的是它用于操作占两个字节(取值范围为0~65535)的寄存器,比如定时器T0和T1。
sbit只占用一个位,和两个值。一般用来给引脚取别名,例如sbit P1_0 = P0^1; 就是定义用符号P1_0来表示P1.0引脚。 需要注意的是,一单用了sbit定义某个变量,这个变量的地址就是确定的了(不能修改了);
w 对于引脚来说,这个0和1是有物理意义的:0代表低电平,1代表高电平。而机器是不懂代码只能识别高低电平。(脑补:这样我们就打通了从代码/软件通往硬件的大路~) 高电平就是5伏正电压,低电平就是0伏,这个是理想值,实际上它也有一个范围......(参考自https://zhidao.baidu.com/question/266456228.html)
w 用法1:sbit 位变量名 = 地址值;
w 用法2:sbit 位变量名 = SFR名称^地址值;
w 用法3:sbit 位变量名 = SFR地址值^变量位地址值;
w 3种用法参考自:https://blog.csdn.net/guzicheng/article/details/7242981
关于8051单片机特殊功能寄存器的说明,可以查阅: https://wenku.baidu.com/view/6f7b242c0975f46526d3e1aa.html?from=search 为防止链接失效,这里给出文件名:《8051,STC89C52单片机特殊功能寄存器》 |
Q2:#define OSC_FREQ 22118400L这句宏命令里的“L”是什么意思?
A2:长整型数字在数字的后面加字母L,如104L,034L等。总结如下:
w 十进制:直接用一般数字来表示,例如123,111,-999等;
w 十六进制:以0x开头,如0xFF,0x01等;
w 长整型:在数字后面加字母L,如104L,034L等
w 浮点型:分为十进制形式和指数形式两种,统一格式为 [±](数)(.数){e[±]数},其中[ ]为可选项,( )表示二者必有其一,{ }十进制不填,指数必填。例如:3.14,-.1,+2;.3e-3,12e+3,6.66e13
w 字符型:用单引号括住括住,例如'a','c'等。对于特殊字符,例如换行符、反斜杠等请参考C语言等教材。
w 字符串:略
Q3:我粘贴了别人的代码,怎么发现没有unit这个类型?
A3:别人的代码只给了函数部分,没有给头文件中的预处理命令。可以在自己的头文件中加入:#typedef unsigned int uint; (后面要加分号),这样就可以用uint类型来代表unsigned int类型了。
Q4:为什么好多变量都是char类型?它不是字符类型吗?怎么可以用来计数?
A4:int在8位的51单片机是占用2个字节,char在占用1个字节,所以说char类型占用空间更小。单片机的存储器很小,尽量不要浪费空间,能用小的就用小的,且一般都用无符号的。
参考:https://zhidao.baidu.com/question/342140311.html
至于它为什么可以计数,因为字符本来就是用二进制表示的,所以当你对char类型的变量赋值时(例如 char a = 'A'),它(a)底层仍然是二进制,将二进制转化为十进制,当然可以用来计数。
Q4.1:51单片机中的char,int,long,float,double各占多少个字节,取值范围多大?
表:Keil uVision4面向51单片机的基本数据类型各种属性一览表
参考:http://blog.sina.com.cn/s/blog_6ac7328f0102uzd2.html
Q5:unsigned char data是什么数据类型?
A5:定义一个变量的格式为:[存储种类] 数据类型 [存储器类型] 变量名表
在定义格式中除了触及类型和变量名表是必要的,其他都是可选项。存储种类有四种:
w auto(自动)、extern(外部)、static(静态)和register(寄存器),默认类型为自动。
存储器类型的说明是指定该变量在C51硬件系统中所使用的存储区域,并在编译是准确定位。如果省略存储器类型,系统则会按编译模式SMALL,COMPACT或LARGE所规定的默认存储器类型去指定变量的存储区域。89C51中的存储器类型有:
w data :可直接寻址的内部数据存储区(128B),访问速度最快;
w idata:间接寻址的内部数据存储区(256B),允许访问全部内存地址;
w bdata:可位寻址内部数据存储区(16B),允许位与字节混合访问;
w pdata:分页的外部数据存储区(256字节),用MOVX @Ri指令访问;
w xdata:外部数据存储区(64KB),用MOVX @A+DPTR指令访问;
w code :程序存储区(64KB),用MOVC @A+DPTR指令访问;
——《51单片机C语言入门教程》,磁动力工作室,第六课 变量
一般需要严格控制变量读取速度的时候用data。例如变量更新速度很快,或者需要很短时间内读取或者修改的变量。一般容量要求大的,但速度并没有太大要求的,放在xdata里面。
如果所有变量都不加这些关键字的话,编译器会自动分配,但编译器的分配方案并不一定是最好的。而且一般都不会非常合理。
参考:https://bbs.csdn.net/topics/360163178
Q6:void timer() interrupt 1 using 2是什么意思?
注:关于“中断”的详细学习放在 第二节:51单片机相关
A5:关键字interrupt表示这是一个中断函数,具体的书写格式为:
void 函数名() interrupt n [using m]
{ do something; }
首先需要注意的是中断函数没有参数传递其无返回值。n表示中断源,m为单片机工作寄存器编号。[using m]为非必须内容。在设计中断时,尽量让中断函数做少量的工作,这样中断服务时间短,系统可以及时的响应其他中断。有些系统如果丢失中断或对中断反应太慢将产生十分严重的后果,这时有充足的时间等待中断是十分重要的。
个中断源,个优先级,可以实现二级中断嵌套。(中断服务进行中再进行一次优先级更高的中断)
所以n的取值为0,1,2,3,4共5个,对应了5种中断源,这5种中断源可以分为三种类型:外部中断,定时器中断,串口中断。
w 0:外部中断0(INT0)
w 1:定时器0(T0)
w 2:外部中断1(INT1)
w 3:定时器1中断(T1)
w 4:串行口中断(RX/TX)
m的取值有0,1,2,3共4个,它涉及到中断的优先权,如果用不到二级中断,using m可以不加,系统会为你自动分配。如果加可能会导致不必要的冲突。
Q7:如何写一个1ms延迟的函数?
A6:在写函数之前首先要认识到,假如采用for循环,则循环一次所花费的时间是多少?这就涉及到单片机深层的概念:机器周期。而单片机的机器周期并不是最小的周期,在计算它之前还要了解一下其他几个周期的定义:
w 晶振频率OSC:单片机的最小系统中有一个晶振,它能够使得CPU跑起来,这个晶振为单片机的CPU提供主频。这个晶振的频率就称为晶振频率(外加频率)。
w 时钟周期Tc:又称为“震荡周期”,它等于晶振频率的导数。这是最基础的周期。
w 机器周期Tm:1机器周期 = 12个震荡周期;单片机复位至少需要两个机器周期的高电平。
w 指令周期Ti:执行一条指令所需的机器周期数。1指令周期 = 1、2、4个机器周期;
Ÿ 一条赋值语句(j = 0)2个机器周期,j是unsign char类型;
Ÿ 一条判断语句(j < 1)4个机器周期,j是unsign char类型;
Ÿ 一条自增/减语句(j++)1个机器周期,j是unsign char类型;
Ÿ 一条空语句(循环体内)1个机器周期。
Ÿ https://zhidao.baidu.com/question/89303563.html
晶振频率OSC |
11.0592 MHz |
12.0000 MHz |
时钟周期 |
9.04225e-5 ms |
8.33333e-5 ms |
机器周期 |
1.08507e-3 ms |
1.00000e-3 ms |
假如采用for循环,例如for(j=X;j>0;j--){ }; 这行代码有X个循环,每次循环有一条判断语句(j>0,4Tm),一条空语句({ },1Tm),一条自减语句(j--,1Tm),略去第一个循环的赋值语句(j=X,2Tm),共6X个机器周期。略去最后一次的判断语句(j=0时,4Tm),若要延迟一秒,只需令6X*Tm = 1,当采用12MHz的晶振时,X ≈ 167。被略去的语句达6Tm,刚好等于一次循环所耗费的时间,所以对X进行X=X-1的修正,最终可得:X ≈ 166。
这里讲的是j为char类型的变量,它最大只能取到255,所以要获得更大的延时,需要用到int类型。前面也学到int类型是16位的,而单片机是8位的,所以这会更加复杂。
参考:https://zhidao.baidu.com/question/89303563.html
下面给的两个延时函数。这里多说一句:我查阅网络资料发现延时1ms的程序不尽相同,甚至相差很大,如果你需要非常准确的延时,推荐你参考正规的教材或采用其他方法比如计时系统。
两种晶振的单片机,延时1ms的函数 |
|
11.0592MHz晶振 |
12MHz晶振 |
void delay_ms(unsigned int i){ |
void delay_ms(unsigned int i){ |
参考:https://blog.csdn.net/feike24/article/details/52357772
这种方法也有很大的缺点:延迟过程中,CPU被占用,无法进行其他任务,导致系统效率降低。延迟时间越长,该缺点便越明显,因此软件延时只适用于短暂延时,或简单项目。
参考:https://www.bilibili.com/video/av15466938/?p=20
Q8:经常看到TH0与TL0,例如TH0 = 0xD8;TL0 = 0xEF;这起什么作用?
A6:从上面的学习可知TH0与TL0是与定时器/计数器有关的SFR寄存器。这两句的含义是给定时/计数器赋初值,寄存器会按固定的时间间隔累加,当寄存器的值达到最大时会触发中断,这时可以利用中断函数进行一系列操作。而计时T就等于时间间隔*(最大值 – 初值)。大概就是这个意思。
二、51单片机相关
Q1:单片机的引脚电压是多少?它的电压是由谁控制的?
A1:单片机的引脚有两种电平:高电平与低电平。高电平的电压与单片机的工作电压有关,一般有5V和3.3V两种。低电平一般为0V。
P0,P1,P2,P3又称为并行I/O端口,还是低电平0;代表红外光被反射并被接收管接收,高电平1代表红外光被外界(黑线等)吸收。
P1称为端口,P1.1称为P1端口的引脚,这两个概念之间的关系就是“整体”与“个体”的关系。
参考:https://wenku.baidu.com/view/39a0bda0700abb68a982fbfa.html
Q2:P3口的8个引脚有哪些复用功能(第二功能),默认开启吗?
A2:
当复用功能没有开启时,P3可以做为普通I/O口使用。一般情况下,复位后第二功能都是关闭的,需要设置对应寄存器才能打开。
Ÿ
P3.0 RXD 串行输入口
Ÿ
P3.1 TXD 串行输出口
Ÿ
P3.2 INT0 外部中断0输入口
Ÿ
P3.3 INT1 外部中断1输入口
Ÿ
P3.4 T0 定时器/计数器0外部时间脉冲输入端
Ÿ
P3.5 T1 定时器/计数器1外部时间脉冲输入端
Ÿ
P3.6 WR 外部数据存储器写脉冲
Ÿ
P3.7 RD 外部数据存储器读脉冲
问题解决1:程序通过USB口无法烧入单片机
我的单片机插座要用一个USB转TTL设备才能从电脑上给单片机烧程序,当时我就在P3.0和P3.1上连了其他模块,结果每次下载都失败。后来我才明白TTL插口是和单片机上的RXD,TXD连着的,下载时是开启了它们的复用功能的。
Q3:我的程序编译后生成的HEX文件超过了8k,烧进单片机不会有问题吗?
A3:HEX文件不只包含了实际的操作指令,还包含了地址代码,这个文件是为了易于下载器的理解。真正下载到单片机上的并不是HEX文件,参考下面的链接,给单片机烧入160+K的HEX文件仍然没有问题。
参考:http://tieba.baidu.com/p/3408654325?red_tag=b2302518919
问题解决2:Keil生成超过8K的HEX文件会报错,提示Target
not created
这是因为你所使用的Keil没有经过注册,需要注册一下就可以生成超过8K的文件了。至于如何注册这里就不多说了。
注意:如果真的是代码量超过所选单片机的容量(STC89C52RC的容量为8K),那么编译器在生产HEX文件时会提示 xxx code limit 之类的。
参考:https://bbs.csdn.net/topics/390011912
Q4:为了写中断程序,我需要详细了解一下中断系统。
A3:CPU在处理某一事件A时,事件B请求CPU迅速去处理,CPU暂时中断当前工作A,转去处理事件B,待CPU将事件B处理完后再返回继续处理A事件,这一过程称为中断。在这之中有几个专业名词需要解释一下:
w
中断发生:事件B请求CPU迅速去处理;
w
中断响应:CPU暂时中断当前工作A;
w
中断服务:CPU转去处理事件B;
w
中断返回:CPU再返回继续处理A事件;
w
断点:程序A被中断的地方;
w
中断源:引起CPU中断的根源。断源能够向CPU提出中断请求;
中断系统的结构如下图所示。
w
第一列解释:INT0:外部中断0,INT1:外部中断1,T0定时器中断0,T1定时器中断0,RX、TX:串口中断(包装在一起的)。中断引起原因如下:
Ÿ
INT0:P3.2引脚低电平或下降沿信号;
Ÿ
T0:定时/计数器0计数回0溢出;
Ÿ
INT1:P3.3引脚低电平或下降沿信号;
Ÿ
T1:定时/计数器1计数0溢出;
Ÿ
串行通信完成一帧数据发送或接受引起中断。
w
第二列TCON解释:该列的第二列代表了五种中断源的中断标志,所谓中断标志就是“中断请求的标志”,CPU要进行中断服务,首先要判断中断请求标志,再判断中断使能标志是否Enable,最后才会响应这个中断。——https://zhidao.baidu.com/question/81735469.html
Ÿ
对于外部中断,当中断到来时(引脚的电平发生变化),硬件会自动将中断标志置为;而对于计时器中断,中断标志的值是可以认为修改,所以可以利用这一点进行人为中断(通过软件/程序),可以达到计数、时钟累加、自检、扫描等目的。
ü
外部中断需要外部条件触发,计时器中断不用。
Ÿ
需要注意的是,无论是机器中断还是人为中断,在中断服务完成后机器并不一定会清除该中断标志位(不同的MCU情况不同),所以为安全起见,我们一般利用程序清除。
Ÿ
外部中断的中断标志前(第一列)各有两个开关,对应了外部中断的两种触发方式:触发;当过渡到低电平的过程)。这两种触发方式有不同的效果,低电平可以持续一段时间,而电平下降却是一瞬间的事,所以两种触发方法在延时效果上不同。
w
第三列IE解释:个开关全部闭合),第一列的5个开关:EX0、ET0、EX1、ET1、ES分别对应了第一列的5个中断源。
Ÿ
如需开启INT0中断,需要将EX0与EA都合上,即EX0=1;EA=1;
w
第四列IP解释:中断优先级控制寄存器。其中IP.7、IP.6与IP.5为保留位,其他位(PS=IP.4, PT1=IP.3,
PX1=IP.2, PT0=IP.1, PX0=IP.0)值为1时表示对应中断源具有高优先级,值为0表示其具有低优先级。
Ÿ
若这5个中断源被设置为同等优先级,则按自然优先级排序依次执行中断服务。如下表所
示:
89C51单片机的中断优先级有三条原则:
1、CPU同时受到几个中断时,首先响应优先级别最高的中断请求。
2、正在进行的中断服务不能被新的同级或低级的中断请求所打断。
3、正在进行的低级中断服务能被高级的中断请求所打断。
、中断源有中断请求;、此中断的中断允许;CPU开总中断(EA=1)。
参考:https://www.bilibili.com/video/av5833218
程序示例1:蜂鸣器滴两次、进入中断服务——数码管显示8,延迟1秒后熄灭。
#include <AT89X51.h> //预处理命令 #define Led #define Buzz // 11.0592M的晶振延迟1ms。这个函数要放在Buzz_didi的上面,否则会报错。 void delay_ms(unsigned int i){ unsigned int for(;i>0;i--) for(j=114;j>0;j--); } // 蜂鸣器发出滴滴声 void Buzz=0; Buzz=1; } void main(){ EA = ET0 = while(1){ Buzz_didi(); // 蜂鸣器滴一次 Buzz_didi(); // 蜂鸣器滴两次 TF0 = 1; // 定时器中断0的中断标志置1 } } // 中断函数一般放在main函数的下面 void LED_Show8() interrupt 1{ Led = delay_ms(1000); // 延迟1秒钟 Led = // TF0 = } |
Q5:为了理解某段程序的作用,我需要详细了解一下计时/计数系统。
A5:单片机中有多个小闹钟(T0,T1;52单片机还有一个T2小闹钟),可以用来计数、定时等。它们的结构图如下
w
定时/计数器0,T0,它的触发引脚为P3.4,计数器为8位寄存器TL0和TH0,用于存放数值,TL0是低八位,TH0是高八位。当低八位计数满了之后会向高八位进一位。对于T1同理。
w
配置寄存器TCON:控制寄存器,控制T0、T1的启动和停止及设置溢出标志,与之相关的sbit由TF1、TR1、TF0、TR0;单片机才有。
Ÿ
TF0、TF1是溢出中断请求标志为,详细参考“本节 Q4:为了写中断程序,我需要详细了解一下中断系统。”
Ÿ
TR0、TR1是运行控制位,TRX=1时,TX开始工作;TRX=0时,TX停止工作。TRX由软件置1或清0,所以可以用软件控制定时/计数器的启动与停止。
w
配置寄存器TMOD:定时/计数器的工作方式寄存器,用来确定工作方式(M0和M1)和功能(GATE和C/T)
Ÿ
C/T:定时器或计数器功能的选择位。计数器的计时间隔为1个机器周期(计数频率为晶振频率的1/12)。所以定时时间T = 计数值N * 机器周期Tm。
Ÿ
GATE:门控位。当GATE=0时,只要用软件使TCON中的TR0或TR1为1,就可以启动定时/计数器工作;当GATE=1时,要用软件是TR0或TR1为1,同时外部中断引脚为高电平时,才能启动定时/计数器工作。我们一般让GATE=0。
T0、T1定时/计数器可以在四种方式下工作,由M0和M1的取值来确定。
方式1,以T0为例:的实质是的由计数脉冲触发的按递增规律(即累加方式)工作的循环位的,由高八位的TH0与第八位的TL0组成。从预先设定的初始值开始,每来一个计数脉冲(时间间隔固定)就加计数器1,当TL0溢出后,对TH0进位,当TH0溢出后,TF0会被硬件置1,从而发出中断请求。
Ÿ
溢出:当计数器的每一位都是1时,对计数器再加1就会溢出,结果就是计数器的每一位都回0。
Ÿ
计数值N = 溢出时计数器的值(65536=2^16) - 计数初值X
Ÿ
当TF0=1时,cpu可以不做响应。学习了后面的中断系统后就会知道,cpu对中断做出响应需要两个判断条件,另外一个就是开启中断使能标志:EA=1;ET0=1;
参考:https://www.bilibili.com/video/av15466938/?p=20
Q6:如何用定时/计数器进行1ms的延迟?
A5:定时器的操作步骤(下面的X代表0或1):
w
选择工作方式(设置M0,M1),这里选用方式1,即M0=1;M1=0;;
w
选择控制方式(设置GATE),一般设GATE=0;;
w
选择定时器还是计数器模式(设置C/T),一般采用定时器模式即C/T=0;;
w
给定时/计数器赋初值(设置THX和TLX),注意结合一下两个公式:
Ÿ
计数值N = 溢出时计数器的值(65536=2^16) - 计数初值X
Ÿ
定时时间T = 计数值N * 机器周期Tm = N*12 / 晶振频率
n
如果频率的单位是MHz(兆赫兹),则时间的单位为us(微秒)
Ÿ
计算出初值进制,个数,个数。或者将其转化为进制,它除以的商为THX,余数为TLX。
w
开启定时器中断(ETX=1;);
w
开总中断(EA=1;);
w
打开计数器(TRX=1)。
程序示例2:定时器的配置
void TimerConfiguration(){ TMOD = TH0 = 0x3C; EA = ET0 = TR0 = } |
程序示例3:蜂鸣器持续滴滴,利用定时器完成Nms的延时,进入中断服务——数码管显示8,延迟1秒后熄灭。
#include #define #define // 11.0592M的晶振延迟1ms。 void unsigned int j; for(;i>0;i--) for(j=114;j>0;j--); } // 初始化定时器0 void // // TMOD TH0 TL0 EA ET0 TR0 } unsigned int void main(){ TimerConfiguration(); while(1){ Buzz=0; delay_ms(100); Buzz=1; delay_ms(200); } } // 中断函数一般放在main函数的下面 void inter_t0() count++; TH0 TL0 //TF0 = 0; // 定时器0的中断标志置0 if(count>1000){ count Led = 0x01; // 数码管显示为8 delay_ms(1000); // 延迟200ms Led } } |
从程序示例1与程序示例3的对比可以得出以下结论:
l
利用定时器中断有两种方法,一种方法只需将定时器0的中断标志置1(TF0 = 1;即程序示例1),另一种方法需要开启定时器0(TR0 = 1;即程序示例3)
l
利用定时器进行延时,这个计时进程与main进程是并行的。
l
无论采用定时器延时中断还是手动中断,中断服务与mian进程总是串行的。
l
改程序的测试结果表明:有时候,当数码管显示8时,蜂鸣器是静音的;而有时候,当数码管显示8时,蜂鸣器在一直鸣响。这说明两点:
n
定时器中断可以中断函数类型的延迟。
n
单片机在服务中断的时候,既定的事实不会发生变化。也就是说若中断发生在蜂鸣器响的中间时刻,则蜂鸣器会一直响下去,直到中断服务返回。
Q7:单片机的有好多特殊寄存器,我需要总结一下他们的名称及用途
A6:关于8051单片机特殊功能寄存器的说明,可以查阅:
https://wenku.baidu.com/view/6f7b242c0975f46526d3e1aa.html?from=search
为防止链接失效,这里给出文件名:《8051,STC89C52单片机特殊功能寄存器》
下面给出第一页的预览图: