51 單片機(jī) C 語言串行雙機(jī)通信的問題
題目:用兩個 89C51 單片機(jī)實(shí)現(xiàn)雙機(jī)通訊來做一個時鐘,A 機(jī)用來產(chǎn)生數(shù)據(jù),B 機(jī)用來顯示。
具體思路是 A 機(jī)產(chǎn)生 a、b、c 的具體值,然后把 abc 傳給 B 機(jī)顯示數(shù)據(jù)。
做而論道回答:先設(shè)計(jì)出來電路,然后再談編程的問題。
追問:不需要你給我完整的程序,我只需要通訊那一塊的程序。端口什么的隨便你用。
我要的是 A 機(jī)發(fā)送 a, b, c 給 B 機(jī),B 機(jī)拿來用。
做而論道按照提問者的要求,寫出了雙方的通信部分,代碼可見該問題的網(wǎng)址:
但是,從后續(xù)的追問來看,提問者顯然還是不知道怎么用。
為此,做而論道就把這個時鐘的全部仿真圖,都畫出來,顯示函數(shù)、定時函數(shù)...,也都編寫出來,供參考。
PROTEUS 仿真電路圖如下:
B 機(jī)(顯示數(shù)據(jù))的程序如下:
//--------------------------------
#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() ? ? ? ?//乙機(jī)主函數(shù)
{
? ? init();
? ? while(1) {
? ? ? display();
? ? }
}
//--------------------------------
recv_abc() ?interrupt 4 ?//乙機(jī)接收
{
? ? 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 機(jī)(產(chǎn)生數(shù)據(jù))的程序如下:
//--------------------------------
#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) ? ?//甲機(jī)發(fā)送
{
? ? SBUF = x; ?while(!TI); ?TI = 0;
}
//--------------------------------
main() ? ? ? ?//甲機(jī)主函數(shù)
{
? ? 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定時中斷函數(shù)
{
? ? 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 文件;再分別用兩個單片機(jī)來裝入。
程序執(zhí)行后,即可顯示出來前面插圖的效果。B 機(jī)的顯示器,每秒更新一次數(shù)據(jù)。
當(dāng)把中間的開關(guān)斷開后,顯示的數(shù)據(jù)便會停頓,不變了。
再把中間的開關(guān)接通后,顯示又會變化,而且數(shù)據(jù)并不受斷開的影響。
這說明,顯示的內(nèi)容,明顯是從 A 機(jī)傳送來的。
//============================================
本題目,需要傳送的數(shù)據(jù)有時、分、秒共三個,這就屬于多字節(jié)的串行通信。
單片機(jī)的串行通信,每次只能傳送一個字節(jié),即 0~255。
多字節(jié)的數(shù)據(jù)傳送,需要制訂協(xié)議。
否則,連續(xù)傳送一個一個的字節(jié),到了接收方,也不知道哪個是時、哪個是分、哪個是秒。
這時,一般要采用 ASCII 碼來傳送。
用 0~9,即代表了一系列有用的數(shù)據(jù)。
再用一個 0~9 之外的符號,當(dāng)做《數(shù)據(jù)頭》,就行了。
本程序,就是以美元符 $ 當(dāng)做數(shù)據(jù)頭。
傳送了 $ 之后,接著就傳送時的十位數(shù)、時的個位數(shù)、分的十位數(shù)、分的位數(shù)、秒的個位數(shù)。
每次發(fā)送數(shù)據(jù),就連續(xù)的發(fā)出七個字節(jié)。
接收方收到了 $ 之后,就把后面再收到的當(dāng)做時、分、秒的十位、個位保存。
當(dāng)收齊了七個字節(jié),就把這后面的六個字節(jié),送去顯示。
本題目要求比較簡單,所以,做而論道編寫的程序,也就沒有包括《檢錯》的部分。
//============================================