找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 4285|回复: 9
收起左侧

单片机双机通信能不能传送数组

[复制链接]
ID:147040 发表于 2016-11-15 11:11 来自手机 | 显示全部楼层 |阅读模式
假如甲单片机发送数组char a [4] ={1,2,3,4};SBUF=a;乙单片机接收char b [4];
该怎么接收,接收完成后a数组与b数组的值是否完全一致?
回复

使用道具 举报

ID:1 发表于 2016-11-15 11:18 | 显示全部楼层
0.png
  1. //============================================
  2. B 机(显示数据)的程序如下:
  3. //--------------------------------
  4. #include<reg52.h>
  5. unsigned char  re_i = 0, r_buf[7] = {0,0,0,0,0,0,0};

  6. unsigned char  a = 13, b = 57, c = 40, d;
  7. //--------------------------------

  8. void delayms(unsigned int xms)

  9. {

  10.     unsigned  int i, j;

  11.     for(i = xms; i > 0; i--)  for(j = 110; j > 0; j--);

  12. }

  13. //--------------------------------

  14. void display()

  15. {   

  16.     char  i, dis[6];

  17.     char code table[] = {

  18.       0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xd8,0x80,0x90};

  19.     char code WEI[] = {1,2,4,8,16,32};



  20.     dis[0] = a / 10; dis[1] = a % 10;

  21.     dis[2] = b / 10; dis[3] = b % 10;

  22.     dis[4] = c / 10; dis[5] = c % 10;



  23.     for (i = 0; i < 6; i++) {

  24.       P0 = table[dis[i]];

  25.       P2 = WEI[i]; delayms(10); P2 = 0;

  26.     }

  27. }

  28. //--------------------------------

  29. init()

  30. {

  31.     PCON = 0;

  32.     SCON = 0x50;

  33.     TMOD = 0x20;

  34.     TH1 = 0xfd;

  35.     TL1 = 0xfd;

  36.     TR1 = 1;

  37.     ES = 1;

  38.     EA = 1;

  39. }

  40. //--------------------------------

  41. main()        //乙机主函数

  42. {

  43.     init();

  44.     while(1) {

  45.       display();

  46.     }

  47. }

  48. //--------------------------------

  49. recv_abc()  interrupt 4  //乙机接收

  50. {

  51.     if (RI) {

  52.       RI = 0;

  53.       d = SBUF;

  54.       if (d == '


  55. )  re_i = 0;

  56.       r_buf[re_i] = d;

  57.       re_i++;

  58.       if (re_i == 7) {

  59.         re_i = 0;

  60.         a = (r_buf[1] - '0') * 10 + (r_buf[2] - '0');

  61.         b = (r_buf[3] - '0') * 10 + (r_buf[4] - '0');

  62.         c = (r_buf[5] - '0') * 10 + (r_buf[6] - '0');

  63.       }

  64.     }

  65. }



  66. //============================================



  67. A 机(产生数据)的程序如下:



  68. //--------------------------------



  69. #include<reg52.h>



  70. unsigned char  a = 13, b = 57, c = 40, d;

  71. bit  sec;

  72. //--------------------------------

  73. init()

  74. {

  75.     PCON = 0;

  76.     SCON = 0x50;

  77.     TMOD = 0x21;

  78.     TH1 = 0xfd;

  79.     TL1 = 0xfd;

  80.     TR1 = 1;

  81.     EA = 1;



  82.     TH0 = 0x4c;

  83.     TR0 = 1;

  84.     ET0 = 1;

  85. }

  86. //--------------------------------

  87. send(unsigned char x)    //甲机发送

  88. {

  89.     SBUF = x;  while(!TI);  TI = 0;

  90. }

  91. //--------------------------------

  92. main()        //甲机主函数

  93. {

  94.     init();

  95.     while(1) {

  96.       if(sec) {

  97.         sec = 0;

  98.         send('


  99. );

  100.         send(a / 10 + '0'); send(a % 10 + '0');

  101.         send(b / 10 + '0'); send(b % 10 + '0');

  102.         send(c / 10 + '0'); send(c % 10 + '0');

  103.     } }

  104. }

  105. //--------------------------------

  106. T0_INT()  interrupt 1  //50ms定时中断函数

  107. {

  108.     TH0 = 0x4c;

  109.     d++;

  110.     if (d >= 2) {  //20

  111.       d = 0;

  112.       sec = 1;

  113.       c++;

  114.       if (c == 60) {

  115.         c = 0;

  116.         b++;

  117.         if (b == 60) {

  118.           b = 0;

  119.           a++;

  120.           if (a == 24) a = 0;

  121.     } } }

  122. }



  123. //============================================



  124. 上述的两个程序,需分别编译,生成不同 HEX 文件;再分别用两个单片机来装入。



  125. 程序执行后,即可显示出来前面插图的效果。B 机的显示器,每秒更新一次数据。



  126. 当把中间的开关断开后,显示的数据便会停顿,不变了。



  127. 再把中间的开关接通后,显示又会变化,而且数据并不受断开的影响。



  128. 这说明,显示的内容,明显是从 A 机传送来的。



  129. //============================================



  130. 本题目,需要传送的数据有时、分、秒共三个,这就属于多字节的串行通信。



  131. 单片机的串行通信,每次只能传送一个字节,即 0~255。



  132. 多字节的数据传送,需要制订协议。



  133. 否则,连续传送一个一个的字节,到了接收方,也不知道哪个是时、哪个是分、哪个是秒。



  134. 这时,一般要采用 ASCII 码来传送。



  135. 用 0~9,即代表了一系列有用的数据。



  136. 再用一个 0~9 之外的符号,当做《数据头》,就行了。



  137. 本程序,就是以美元符 $ 当做数据头。



  138. 传送了 $ 之后,接着就传送时的十位数、时的个位数、分的十位数、分的位数、秒的个位数。



  139. 每次发送数据,就连续的发出七个字节。



  140. 接收方收到了 $ 之后,就把后面再收到的当做时、分、秒的十位、个位保存。



  141. 当收齐了七个字节,就把这后面的六个字节,送去显示。
复制代码



回复

使用道具 举报

ID:111634 发表于 2016-11-15 12:22 | 显示全部楼层
为什么不可以?
回复

使用道具 举报

ID:147040 发表于 2016-11-15 13:22 来自手机 | 显示全部楼层
admin 发表于 2016-11-15 11:18

简直帅的掉渣,我基本懂了。我还有两个比较2的问题,
第一个,  send(a / 10 + '0');   a = (r_buf[1] - '0')        这里的+'0'  和  -'0'  是什么意思?
第二个,如果SBUF=120;那么发送是把120一块发送过去,还是分三次‘1’,‘2’,‘0’发送。
如果我定义char a=120;SBUF=a;那么有是分几次发送。

再次感谢!
回复

使用道具 举报

ID:147040 发表于 2016-11-15 13:23 来自手机 | 显示全部楼层
zl2168 发表于 2016-11-15 12:22
为什么不可以?

菜鸟不太懂(;`O)o
回复

使用道具 举报

ID:148019 发表于 2016-11-15 13:52 | 显示全部楼层
应该是不一致的
回复

使用道具 举报

ID:81808 发表于 2016-11-15 23:19 | 显示全部楼层
lucky666 发表于 2016-11-15 13:22
简直帅的掉渣,我基本懂了。我还有两个比较2的问题,
第一个,  send(a / 10 + '0');   a = (r_buf[1] - ...

第一个,是数字与ASCII转换;第二个,分一次发送,改善的是120的16进制,就是0x78,二进制是_0111_1000
回复

使用道具 举报

ID:147040 发表于 2016-11-16 00:05 来自手机 | 显示全部楼层
yanjibao 发表于 2016-11-15 23:19
第一个,是数字与ASCII转换;第二个,分一次发送,改善的是120的16进制,就是0x78,二进制是_0111_1000

恩,感谢感谢,又学到了!
回复

使用道具 举报

ID:148258 发表于 2016-11-16 15:07 | 显示全部楼层
当然可以了
回复

使用道具 举报

ID:153758 发表于 2016-12-10 19:53 来自手机 | 显示全部楼层
用$当数据头,可是在keil里输入$,编译会出错。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表