隔了两天没有写笔记了,也不算是偷懒吧,因为这两天写的程序也不少的,只是有些问题没有弄清楚,所以没有写笔记,今天基本上弄懂了,但还是有一些地方处理不好,还是把它留到日后处理吧,感觉这些问题不是我一时半会能够搞定的。
这两天主要是学习了51单片机的计时器的使用。分别尝试了查询法计时和中断法计时。
其中查询法计时我自己编写程序时检测完 TF0 口数据为1时(数据溢出),时间到。于是开始执行相应的指令,但忘记用clr tf0来使tf0口清零,以置后续计时出错。参考书本后发现有jbc这样的指令,可以在检测到tf0为1时跳转且把tf0位清零。免去clr指令。
下面说一说重点。数字电子时钟。
这个程序一共有200多行,第一次写那么长的程序,但其实很多地方是重复的,只要稍稍修改一下引脚就可以了。编写这个程序让我学会了编写程序的时候应先做一个类似的程序,然后其它稍加改进,就可以完成,一个模块一个模块地去处理。写多长也不觉得可怕了。
以下是一些心得:
使用keil软件编译时,保护现场所用的堆栈指令push 和pop指令不能直接用push a 或pop a ,而要写成push acc
和pop acc,这里就涉及了acc与a的区别。虽然acc与a都是代表累加器a,但实际又有区别
*********************************************
如下:
INC A 是寄存器寻址
INC A 是单字节指令,转换成机器码是:04H,机器码中没有明确指出操作数,隐含操作
数是累加器A,所以把这种方式寻址叫寄存器寻址。
INC ACC是直接寻址
INC ACC 是双字节指令,转换成机器码是:05H,E0H。机器码中包含了累加器A的地址E0H
这个 E0H 可以换成其他直接地址,所以这种方式寻址叫直接寻址。
PUSH 和 POP 指令只支持直接寻址,所以不能用 A
PUSH ACC 机器码:C0H E0H
POP ACC 机器码:D0H E0H
*******************************
所以push和pop指令都要用acc。还有在中断中调用push和pop要成对出现,而且在有不同出口的中断子程序中,应在出口前都要加上pop指令,否则在实际中整个程序反而因为“保护现场”而出错。
我这个数字电子时钟还需要按键处理来调节时间,所以我分别对“时,分,秒”都设置了两个按键来处理,一个是 1分钟(秒,时)1分钟(秒,时)地加,另一个按键是 10分钟(秒,时)10分钟(秒,时)地加。因为我显示子程序中设置了CJNE SECOND,#60,FANHUI 用于到60或24时就清零。但这个指令只能针对1+1+1+1这样的方式,如果从54分时,再按一下加10分钟,这时就会显示64分,明显错误,但CJNE 只能用于检测是否等于60,不能判定大小 。这时我才发现汇编的指令中没有比较大小的指令。在网上发贴咨询后终于找到了方法。
*******************************
51单片机中有一个cy寄存器,当使用cjne指令时,如:cjne a,#60,leb 如果累加器中a小于或等于60时,cy寄存器中的值就会变为 cy=1 ,如果是等于60就执行下一个指令,如果是小于就跳到leb中执行。
当a中的数值大于60时,cy就会变为cy=0。并跳到标号leb中执行。所以就可以通过检测cy中的数值来判定a中的数据是大于60还是小于60了。从而实现数的大小比较。
***************************
数的大小比较告一段落,于是我开始修改程序,添加限制数不能超过60的指令,每个按键者添加上比较指令后,出现跳转超出范围的编译错误如sjmp跳转超出范围,jnb跳转超出范围。就是这个问题,目前的我还没
方法处理,搞不懂jnb之类的指令跳转范围。但我换了一个方式,把比较指令从检测按键中整合到了输出前检测中,完善了这个时钟程序。(花了我几个小时去完善)。
还有一个收获:
汇编节约空间方法:
在完善程序的过程中我发现了一个节约空间的方法,就是把数据相同的数先用mov ri,#xx
然后什么时候需要就把用 mov xx,ri ,即用寄存器中的数代替实际的数字,这样每条就能省下一个字节的空间哦。
SECOND EQU 20H ;***************内存地址命名 ****************
MINUTE EQU 21H ;***************内存地址命名 ****************
HOUR EQU 22H ;***************内存地址命名 ****************
ORG 0000H
SJMP START
ORG 000BH
SJMP COUNT ;***************中断子程序存放地址 ****************
ORG 0030H
;***************数据初始化****************
START:
MOV SP,#060H ;***************指针初始化 ****************
MOV P0,#0FFH ;***************高电平,熄灭 ****************
MOV P2,#0FFH ;***************高电平,熄灭 ****************
MOV P1,#0FFH ;***************高电平,熄灭 ****************
MOV R0,#00H ;*************** r0置零 ****************
MOV R3,#00H ;*************** r3置零,用于其它变量清零 ****************
MOV R5,#10 ;*************** r5置10,用于按键累加及b的赋值 ****************
MOV DPTR,#TAB ;*************** 查表指针初始化 ****************
MOV TH0,#03CH ;*************** 计时高8位 ****************
MOV TL0,#0B0H ;******计时低8位 ,与高8位联合,赋计时器初始值****************
MOV TMOD,#00000001B ;*************** 计时器模式1 ****************
MOV IE,#10000010B ;*************** 中断允许 ****************
SETB TR0 ;*************** 计时器开始计时 ****************
;***************无限循环的主程序****************
MAIN:
CALL DISPLAY
CALL KEY
SJMP MAIN
;***************中断触发的计时处理程序****************
COUNT:
PUSH ACC ;***************入栈,保护累加器****************
INC R0 ;***************一个50ms结束,计数器r0加1 ****************
MOV TH0,#03CH ;***************计时器重置****************
MOV TL0,#0B0H ;***************计时器重置****************
CJNE R0,#20,FANHUI ;*************** r0不等于20(不足1秒),跳转继续计时****************
MOV R0,#0H ;*************** 1秒时间到,r0重置****************
INC SECOND ;*************** 1秒时间到,秒位加1 ****************
MOV A,SECOND ;***************秒位数据转移至累加器****************
CJNE A,#60,FANHUI ;***************判定秒位是否等于60,不等不处理,返回****************
MOV SECOND,R3 ;***************等于60,秒位清零****************
INC MINUTE ;***************等于60秒,分位加1 ****************
MOV A,MINUTE ;***************分位数据转移****************
CJNE A,#60,FANHUI ;***************判定分位是否等于60,不等不处理,返回*****
INC HOUR ;***************等于60,分位清零****************
MOV MINUTE,R3 ;***************等于60分,时位加1 ****************
MOV A,HOUR ;***************时位数据转移****************
CJNE A,#24,FANHUI ;***************判定是否等于24,不是不处理,返回 ****************
MOV HOUR,R3 ;***************等于24时,时位清零 ****************
FANHUI: POP ACC ;***************出栈,还原累加器数据****************
RETI ;***************中断子程序返回出口****************
;***************显示子程序****************
DISPLAY:
;***************秒位输出****************
SECONDDIS:
MOV A,SECOND ;***************数据转移,为了比较****************
MOV B,R5 ;*******************小于60,继续执行***********
DIV AB ;***************数字分隔(除法)处理****************
MOVC A,@A+DPTR ;***************十位查表****************
MOV P0,A ;***************十位数据输出****************
CLR P1.6 ;***************控制总线控制p1.6数码管亮****************
CALL DELAY ;***************点亮延时****************
SETB P1.6 ;***************熄灭****************
MOV A,B ;***************个位数据转移****************
MOVC A,@A+DPTR ;***************查表****************
MOV P0,A ;***************个位数据输出****************
CLR P1.7 ;***************控制总线控制p1.7数码管亮****************
CALL DELAY ;***************点亮延时****************
SETB P1.7 ;***************熄灭*
『本文转载自网络,版权归原作者所有,如有侵权请联系删除』