ORG 0200H ;主程序从0200H开始 MAIN: MOV TMOD,#01H ;使用定时器T0,工作方式1
MOV TH0, #3CH ;置初T0值50ms MOV TL0, #0B0HH
MOV R0,#0 ;用于存1秒的计数次数 SETB EA ;开中断总允许 SETB ET0 ;允许T0中断 SETB TR0 ;启动计时
LOOP: LCALL Display ;循环调用显示子程序
SJMP LOOP
TAB: DB 3FH,06H, 5BH, 4FH, 66H, 6DH, 7DH, 07H, 7FH, 6FH ;0~9的共阴极段码 DISPLAY: ;显示子程序(显示分钟)
MOV DPTR, #TAB ;将表首地址赋给DPTR
MOV A, 30H ;将30H中存放的分计数赋给累加器A MOV B, #0AH ;将10赋给累加器B
DIV AB ;分计数除以10得十位数放在A中,个位数放在B中 MOVX A, @A+DPTR ;查表得十位数的`显示段码 MOV SBUF, A ;发送十位数
L1: JBC TI, L2 ;判是否发送完,未发完循环等待,若发完则转L2
SJMP L1
L2: MOV A, B ;将个位数赋给累加器A
MOVX A, @ A+DPTR ;查表得个`位数的显示段码 MOV SBUF, A ;发送个位数
L3: JBC TI, L4 ;判是否发送完,未发完循环等待,若发完则转L4
SJMP L3
L4: RET ;子程序返回 INTERRUPT: ;中断服务子程序
INC R0 ;每中断一次(50ms)加1
CJNE R0, #20, L5 ;判是否中断20次,若不是则转L5中断返回,若是则顺序执行 CLR R0 ;到1秒钟,清R0
INC R1 ;秒计数加1
CJNE R1, #60, L5 ;判秒是否计满60次,若未满则转L5中断返回,若满则顺序执行 CLR R1 ;如果秒计满60,将秒计数单元内容清0 INC 30H ;分钟计数单元30H内容加1
CJNE 30H, #99, L5 ;判分是否计满99次,若未满则转L5中断返回,若满则顺序执行 CLR R1 ;如果分计满99,将秒计数单元内容清0 CLR 30H ;同时将分计数单元内容清0 L5: MOV TH0, #3CH ; 定时器重新赋初值
MOV TL0, #0B0HH RETI ;中断返回 C语言参考程序:
#include
unsigned char Tab[ ]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};
//数组Tab放0~9的共阴极字段码
unsigned char int_time; //设中断次数计数变量 unsigned char second; //秒计数变量 unsigned char minute; //分钟计数变量 void delay(void) //延时函数 {
unsigned char n,j; //设计数循环变量 }
void DisplayMinute(unsigned char m) //显示函数
{ unsigned char ge, si; //定义变量ge、si,用于存放个位、十位 si=Tab[m/10]; //计算出十位的值,查表转换成相应的段码送变量si SBUF=si; //发送显示十位 while(TI==0); //等待发送完毕 TI=0; //发送完后清中断标志
ge=Tab[m]; //计算出十位的值,查表转换成相应的段码送变量ge for(j=0;j<200;j++); //通过循环延时 for(n=0;n<200;n++); //通过循环延时
SBUF=ge; //发送显示个位 while(TI==0); //等待发送完毕 TI=0; //发送完后清中断标志
delay(); //调延时函数,是数码管显示有一定的亮度 delay(); }
void main(void) //主函数 {
TMOD=0x01; //使用定时器T0,工作方式1 EA=1; //开中断总允许 ET0=1; //允许T0中断
TH0=(65536-46083)/256; //定时器高八位赋初值(50ms) TL0=(65536-46083)%6; //定时器低八位赋初值 TR0=1; //启动计时
PCON=0x00; //置SMOD=0
SCON=0x00; //串行口工作在方式0 while(1) //无限循环体
{
DisplayMinute(minute); //调用分钟显示子程序 } }
void interserve(void ) interrupt 1 //计数器T0中断函数 {
int_time++; //每中断一次(50ms)加1
if(int_time==20) // 50ms记20次为1秒(仿真时可将次数改小,减少等待时间)。 {
int_time=0; //中断计数变量清0 second++; //秒计数变量加1
}
if(second==60) //判是否到了60秒(仿真时可将次数改小,减少等待时间)。
{
second=0; //如果秒计满60,将秒计数变量清0 minute++; //分钟计数变量加1 }
if(minute==99) //判分钟是否等于99
{ minute=0; //如果分钟计满99,将分钟计数变量清0秒计数变量清0 second=0; //将秒计数变量清0 }
TH0=(65536-46083)/256; //定时器重新赋初值
TL0=(65536-46083)%6;
}
2.在Proteus下,仿真实现例6-2内容。
答:电路如图6-8所示,试编制程序输入K1~K8的状态信息,并存入内部RAM 40H。
图6-8 串行通讯方式0应用
4014 是一个并入串出转换芯片,Q8端为串行数据输出端,CLK为时钟脉冲输入端,P/S为操作控制端,P/S=1:锁存并行输入数据,P/S=0:允许串行移位操作。
要完成题目的要求,应先将开关状态锁存,然后串行传送给单片机。 在Proteus下画出电路图。 汇编语言参考程序如下:
ORG 0000H ;上电后程序从00000H开始,在0000H单元存放转移指令
LJMP KIN ;转移到主程序
ORG 0100H ;主程序从0100H开始 KIN: MOV SCON,#00H ;设定串行口为方式0 CLR ES ;禁止串行中断 SETB P1.0 ;锁存并行输入数据 CLR P1.0 ;允许串行移位操作
SETB REN ;允许并启动接收(TXD发送移位脉冲) JNB RI,$ ;等待接收完毕 MOV 40H,SBUF ;存入K1~K8状态数据 SJMP $ ;循环等待 END ;汇编结束 C语言参考程序:
# include
sbit P1_0=P1^0; //定义P1.0口,程序中用P1_0代替P1.0。 void main() //主函数 {
unsigned char i; //定义变量i,将接收到的数据放到变量i中。 P1_0=1; //锁存并行输入数据 P1_0=0; //允许串行移位操作 SCON=0x00; //设定串行口为方式0 while (!RI) {;} //等待接收完毕 RI=0; //清中断标志 i=SBUF; //存入K1~K8状态数据 ?? }
百度搜索“77cn”或“免费范文网”即可找到本站免费阅读全部范文。收藏本站方便下次阅读,免费范文网,提供经典小说综合文库第06章 单片机串行通信系统 单片机原理与应用-基于实例驱动和Pro(2)在线全文阅读。
相关推荐: