题目:用两个 89C51 单片机实现双机通讯来做一个时钟,A 机用来产生数据,B 机用来显示。
具体思路是 A 机产生 a、b、c 的具体值,然后把 abc 传给 B 机显示数据。
做而论道回答:先设计出来电路,然后再谈编程的问题。
追问:不需要你给我完整的程序,我只需要通讯那一块的程序。端口什么的随便你用。
我要的是 A 机发送 a, b, c 给 B 机,B 机拿来用。
做而论道按照提问者的要求,写出了双方的通信部分,代码可见该问题的网址:
http://zhidao.baidu.com/question/1731754753643347587.html
但是,从后续的追问来看,提问者显然还是不知道怎么用。
为此,做而论道就把这个时钟的全部仿真图,都画出来,显示函数、定时函数...,也都编写出来,供参考。
PROTEUS 仿真电路图如下:
图片链接:
http://xiangce.baidu.com/picture/detail/8d7c506b7ade7f0feb49f353e22d81dfef5453dc
//============================================
B 机(显示数据)的程序如下:
//--------------------------------
#include
unsigned char re_i = 0, r_buf[7] = {0,0,0,0,0,0,0};
unsigned char a = 13, b = 57, c = 40, d;
//--------------------------------
void delayms(unsigned int xms)
{
unsigned int i, j;
for(i = xms; i > 0; i--) for(j = 110; j > 0; j--);
}
//--------------------------------
void display()
{
char i, dis[6];
char code table[] = {
0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xd8,0x80,0x90};
char code WEI[] = {1,2,4,8,16,32};
dis[0] = a / 10; dis[1] = a % 10;
dis[2] = b / 10; dis[3] = b % 10;
dis[4] = c / 10; dis[5] = c % 10;
for (i = 0; i < 6; i++) {
P0 = table[dis[i]];
P2 = WEI[i]; delayms(10); P2 = 0;
}
}
//--------------------------------
init()
{
PCON = 0;
SCON = 0x50;
TMOD = 0x20;
TH1 = 0xfd;
TL1 = 0xfd;
TR1 = 1;
ES = 1;
EA = 1;
}
//--------------------------------
main() //乙机主函数
{
init();
while(1) {
display();
}
}
//--------------------------------
recv_abc() interrupt 4 //乙机接收
{
if (RI) {
RI = 0;
d = SBUF;
if (d == '$') re_i = 0;
r_buf[re_i] = d;
re_i++;
if (re_i == 7) {
re_i = 0;
a = (r_buf[1] - '0') * 10 + (r_buf[2] - '0');
b = (r_buf[3] - '0') * 10 + (r_buf[4] - '0');
c = (r_buf[5] - '0') * 10 + (r_buf[6] - '0');
}
}
}
//============================================
A 机(产生数据)的程序如下:
//--------------------------------
#include
unsigned char a = 13, b = 57, c = 40, d;
bit sec;
//--------------------------------
init()
{
PCON = 0;
SCON = 0x50;
TMOD = 0x21;
TH1 = 0xfd;
TL1 = 0xfd;
TR1 = 1;
EA = 1;
TH0 = 0x4c;
TR0 = 1;
ET0 = 1;
}
//--------------------------------
send(unsigned char x) //甲机发送
{
SBUF = x; while(!TI); TI = 0;
}
//--------------------------------
main() //甲机主函数
{
init();
while(1) {
if(sec) {
sec = 0;
send('$');
send(a / 10 + '0'); send(a % 10 + '0');
send(b / 10 + '0'); send(b % 10 + '0');
send(c / 10 + '0'); send(c % 10 + '0');
} }
}
//--------------------------------
T0_INT() interrupt 1 //50ms定时中断函数
{
TH0 = 0x4c;
d++;
if (d >= 2) { //20
d = 0;
sec = 1;
c++;
if (c == 60) {
c = 0;
b++;
if (b == 60) {
b = 0;
a++;
if (a == 24) a = 0;
} } }
}
//============================================
上述的两个程序,需分别编译,生成不同 HEX 文件;再分别用两个单片机来装入。
程序执行后,即可显示出来前面插图的效果。B 机的显示器,每秒更新一次数据。
当把中间的开关断开后,显示的数据便会停顿,不变了。
再把中间的开关接通后,显示又会变化,而且数据并不受断开的影响。
这说明,显示的内容,明显是从 A 机传送来的。
//============================================
本题目,需要传送的数据有时、分、秒共三个,这就属于多字节的串行通信。
单片机的串行通信,每次只能传送一个字节,即 0~255。
多字节的数据传送,需要制订协议。
否则,连续传送一个一个的字节,到了接收方,也不知道哪个是时、哪个是分、哪个是秒。
这时,一般要采用 ASCII 码来传送。
用 0~9,即代表了一系列有用的数据。
再用一个 0~9 之外的符号,当做《数据头》,就行了。
本程序,就是以美元符 $ 当做数据头。
传送了 $ 之后,接着就传送时的十位数、时的个位数、分的十位数、分的位数、秒的个位数。
每次发送数据,就连续的发出七个字节。
接收方收到了 $ 之后,就把后面再收到的当做时、分、秒的十位、个位保存。
当收齐了七个字节,就把这后面的六个字节,送去显示。
本题目要求比较简单,所以,做而论道编写的程序,也就没有包括《检错》的部分。
//============================================