<meter id="pryje"><nav id="pryje"><delect id="pryje"></delect></nav></meter>
          <label id="pryje"></label>

          新聞中心

          詳解RS485通信

          作者: 時(shí)間:2016-12-16 來(lái)源:網(wǎng)絡(luò) 收藏

            在進(jìn)行 RS485 實(shí)驗(yàn)中,我們通信用的引腳必須是 P3.0 和 P3.1,此外還有一個(gè)方向控制引腳,我們使用杜邦線將其連接到 P1.7 上去。RS485 的另外一端,大家可以使用一個(gè) USB轉(zhuǎn) RS485 模塊,用雙絞線把開(kāi)發(fā)板和模塊上的 A 和 B 分別對(duì)應(yīng)連起來(lái),USB 那頭插入電腦,然后就可以進(jìn)行通信了。

            學(xué)習(xí)了第 13 章實(shí)用的串口通信方法和程序后,做這種串口通信的方法就很簡(jiǎn)單了,基本是一致的。我們使用實(shí)用串口通信例程的思路,做了一個(gè)簡(jiǎn)單的程序,通過(guò)串口調(diào)試助手下發(fā)任意個(gè)字符,單片機(jī)接收到后在末尾添加“回車+換行”符后再送回,在調(diào)試助手上重新顯示出來(lái),先把程序貼出來(lái)。

            程序中需要注意的一點(diǎn)是:因?yàn)槠匠6际菍?MAX485 設(shè)置為接收狀態(tài),只有在發(fā)送數(shù)據(jù)的時(shí)候才將 MAX485 改為發(fā)送狀態(tài),所以在 UartWrite()函數(shù)開(kāi)頭將 MAX485 方向引腳拉高,函數(shù)退出前再拉低。但是這里有一個(gè)細(xì)節(jié),就是單片機(jī)的發(fā)送和接收中斷產(chǎn)生的時(shí)刻都是在停止位的一半上,也就是說(shuō)每當(dāng)停止位傳送了一半的時(shí)候,RI 或 TI 就已經(jīng)置位并且馬上進(jìn)入中斷(如果中斷使能的話)函數(shù)了,接收的時(shí)候自然不會(huì)存在問(wèn)題,但發(fā)送的時(shí)候就不一樣了:當(dāng)緊接著向 SBUF 寫入一個(gè)字節(jié)數(shù)據(jù)時(shí),UART 硬件會(huì)在完成上一個(gè)停止位的發(fā)送后,再開(kāi)始新字節(jié)的發(fā)送,但如果此時(shí)不是繼續(xù)發(fā)送下一個(gè)字節(jié),而是已經(jīng)發(fā)送完畢了,要停止發(fā)送并將 MAX485 方向引腳拉低以使 MAX485 重新處于接收狀態(tài)時(shí)就有問(wèn)題了,因?yàn)檫@時(shí)候最后的這個(gè)停止位實(shí)際只發(fā)送了一半,還沒(méi)有完全完成,所以就有了 UartWrite()函數(shù)內(nèi)DelayX10us(5)這個(gè)操作,這是人為的增加了 50us 的延時(shí),這 50us 的時(shí)間正好讓剩下的一半停止位完成,那么這個(gè)時(shí)間自然就是由通信波特率決定的了,為波特率周期的一半。

            /****************************RS485.c 文件程序源代碼*****************************/

          view plaincopy to clipboardprint?
          1. #include
          2. #include
          3. sbitRS485_DIR=P1^7;//RS485方向選擇引腳
          4. bitflagFrame=0;//幀接收完成標(biāo)志,即接收到一幀新數(shù)據(jù)
          5. bitflagTxd=0;//單字節(jié)發(fā)送完成標(biāo)志,用來(lái)替代TXD中斷標(biāo)志位
          6. unsignedcharcntRxd=0;//接收字節(jié)計(jì)數(shù)器
          7. unsignedcharpdatabufRxd[64];//接收字節(jié)緩沖區(qū)
          8. externvoidUartAction(unsignedchar*buf,unsignedcharlen);
          9. /*串口配置函數(shù),baud-通信波特率*/
          10. voidConfigUART(unsignedintbaud){
          11. RS485_DIR=0;//RS485設(shè)置為接收方向
          12. SCON=0x50;//配置串口為模式1
          13. TMOD&=0x0F;//清零T1的控制位
          14. TMOD|=0x20;//配置T1為模式2
          15. TH1=256-(11059200/12/32)/baud;//計(jì)算T1重載值
          16. TL1=TH1;//初值等于重載值
          17. ET1=0;//禁止T1中斷
          18. ES=1;//使能串口中斷
          19. TR1=1;//啟動(dòng)T1
          20. }
          21. /*軟件延時(shí)函數(shù),延時(shí)時(shí)間(t*10)us*/
          22. voidDelayX10us(unsignedchart){
          23. do{
          24. _nop_();
          25. _nop_();
          26. _nop_();
          27. _nop_();
          28. _nop_();
          29. _nop_();
          30. _nop_();
          31. _nop_();
          32. }while(--t);
          33. }
          34. /*串口數(shù)據(jù)寫入,即串口發(fā)送函數(shù),buf-待發(fā)送數(shù)據(jù)的指針,len-指定的發(fā)送長(zhǎng)度*/
          35. voidUartWrite(unsignedchar*buf,unsignedcharlen){
          36. RS485_DIR=1;//RS485設(shè)置為發(fā)送
          37. while(len--){//循環(huán)發(fā)送所有字節(jié)
          38. flagTxd=0;//清零發(fā)送標(biāo)志
          39. SBUF=*buf++;//發(fā)送一個(gè)字節(jié)數(shù)據(jù)
          40. while(!flagTxd);//等待該字節(jié)發(fā)送完成
          41. }
          42. DelayX10us(5);//等待最后的停止位完成,延時(shí)時(shí)間由波特率決定
          43. RS485_DIR=0;//RS485設(shè)置為接收
          44. }
          45. /*串口數(shù)據(jù)讀取函數(shù),buf-接收指針,len-指定的讀取長(zhǎng)度,返回值-實(shí)際讀到的長(zhǎng)度*/
          46. unsignedcharUartRead(unsignedchar*buf,unsignedcharlen){
          47. unsignedchari;
          48. //指定讀取長(zhǎng)度大于實(shí)際接收到的數(shù)據(jù)長(zhǎng)度時(shí),
          49. //讀取長(zhǎng)度設(shè)置為實(shí)際接收到的數(shù)據(jù)長(zhǎng)度
          50. if(len>cntRxd){
          51. len=cntRxd;
          52. }
          53. for(i=0;i
          54. *buf++=bufRxd[i];
          55. }
          56. cntRxd=0;//接收計(jì)數(shù)器清零
          57. returnlen;//返回實(shí)際讀取長(zhǎng)度
          58. }
          59. /*串口接收監(jiān)控,由空閑時(shí)間判定幀結(jié)束,需在定時(shí)中斷中調(diào)用,ms-定時(shí)間隔*/
          60. voidUartRxMonitor(unsignedcharms){
          61. staticunsignedcharcntbkp=0;
          62. staticunsignedcharidletmr=0;
          63. if(cntRxd>0){//接收計(jì)數(shù)器大于零時(shí),監(jiān)控總線空閑時(shí)間
          64. if(cntbkp!=cntRxd){//接收計(jì)數(shù)器改變,即剛接收到數(shù)據(jù)時(shí),清零空閑計(jì)時(shí)
          65. cntbkp=cntRxd;
          66. idletmr=0;
          67. }else{//接收計(jì)數(shù)器未改變,即總線空閑時(shí),累積空閑時(shí)間
          68. if(idletmr<30){//空閑計(jì)時(shí)小于30ms時(shí),持續(xù)累加
          69. idletmr+=ms;
          70. if(idletmr>=30){//空閑時(shí)間達(dá)到30ms時(shí),即判定為一幀接收完畢
          71. flagFrame=1;//設(shè)置幀接收完成標(biāo)志
          72. }
          73. }
          74. }
          75. }else{
          76. cntbkp=0;
          77. }
          78. }
          79. /*串口驅(qū)動(dòng)函數(shù),監(jiān)測(cè)數(shù)據(jù)幀的接收,調(diào)度功能函數(shù),需在主循環(huán)中調(diào)用*/
          80. voidUartDriver(){
          81. unsignedcharlen;
          82. unsignedcharpdatabuf[40];
          83. if(flagFrame){//有命令到達(dá)時(shí),讀取處理該命令
          84. flagFrame=0;
          85. len=UartRead(buf,sizeof(buf)-2);//將接收到的命令讀取到緩沖區(qū)中
          86. UartAction(buf,len);//傳遞數(shù)據(jù)幀,調(diào)用動(dòng)作執(zhí)行函數(shù)
          87. }
          88. }
          89. /*串口中斷服務(wù)函數(shù)*/
          90. voidInterruptUART()interrupt4{
          91. if(RI){//接收到新字節(jié)
          92. RI=0;//清零接收中斷標(biāo)志位
          93. //接收緩沖區(qū)尚未用完時(shí),保存接收字節(jié),并遞增計(jì)數(shù)器
          94. if(cntRxd
          95. bufRxd[cntRxd++]=SBUF;
          96. }
          97. }
          98. if(TI){//字節(jié)發(fā)送完畢
          99. TI=0;//清零發(fā)送中斷標(biāo)志位
          100. flagTxd=1;//設(shè)置字節(jié)發(fā)送完成標(biāo)志
          101. }
          102. }



          關(guān)鍵詞: RS485通

          評(píng)論


          技術(shù)專區(qū)

          關(guān)閉
          看屁屁www成人影院,亚洲人妻成人图片,亚洲精品成人午夜在线,日韩在线 欧美成人 (function(){ var bp = document.createElement('script'); var curProtocol = window.location.protocol.split(':')[0]; if (curProtocol === 'https') { bp.src = 'https://zz.bdstatic.com/linksubmit/push.js'; } else { bp.src = 'http://push.zhanzhang.baidu.com/push.js'; } var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(bp, s); })();