找回密码
 立即注册
楼主: qepuemc

485通信错误用STC单片机和485芯片做USB转串口有错误

[复制链接]
  • 打卡等级:常住居民II
  • 打卡总天数:92
  • 最近打卡:2025-12-21 00:43:30

17

主题

150

回帖

1521

积分

金牌会员

积分
1521
发表于 2025-10-8 19:03:12 | 显示全部楼层
没有数据乱码,就是分包的问题,加大串口接收软件的超时时间可勉强解决。

根本原因是程序问题,发送字节之间或接收字节之间有不确定的间隔时间,
网上买的USB转串口是采用专用硬件芯片,是实时处理的,发送时字节是连续的,接收的字节也立即上传到电脑端。
你做的这个,程序处理有很多查询操作,中断有延时,还有缓冲区来回复制,没有双缓冲。因此字节流是间断的。
字节之间延时稍微一大,就被判定为分包结束了。
回复

使用道具 举报 送花

  • 打卡等级:以坛为家II
  • 打卡总天数:596
  • 最近打卡:2025-12-20 09:04:18

51

主题

397

回帖

860

积分

高级会员

积分
860
发表于 2025-10-9 08:04:25 | 显示全部楼层
666.png

这个是两个买的。同样有问题。所以说这个还是硬件有问题。我想到一个方法不过还没有测试。如果测试成功。这个问题就算是真正解决了。
以前就是因为看到网上买的也是这样的问题。所以就认为有这个问题。其实整好了是没有的。
回复

使用道具 举报 送花

  • 打卡等级:以坛为家II
  • 打卡总天数:587
  • 最近打卡:2025-12-20 08:41:52
已绑定手机

49

主题

2369

回帖

8424

积分

论坛元老

积分
8424
发表于 2025-10-9 09:27:08 | 显示全部楼层
截图202510090920453480.jpg

串口工具上面一般都有超时时间可以设置,两个数据之间间隔时间超过这个时间就会分行显示:
截图202510090926391816.jpg
回复

使用道具 举报 送花

  • 打卡等级:以坛为家II
  • 打卡总天数:596
  • 最近打卡:2025-12-20 09:04:18

51

主题

397

回帖

860

积分

高级会员

积分
860
发表于 2025-10-11 10:17:27 来自手机 | 显示全部楼层
乘风飞扬 发表于 2025-10-9 09:27
串口工具上面一般都有超时时间可以设置,两个数据之间间隔时间超过这个时间就会分行显示:
...

非常非常感谢你的回复。但是事情远远没有这么简单
第一件事就是论响应时间。单片机的响应时间,有时候或者说大多数时候可以搞到一毫秒之内。当然,这样可能要建立在波特率在嗯9600或者115200的时候,或者说115200的时候会更加真实。
我试过用stc的官方程序,然后做了修改之后嗯,会有这个效果,也就是说串口工具发过去会瞬间发回来。但是这是在USB转ttl的操作下实现的。如果是用USB,转485再去到485芯片,再去到stc单片机,目前我用的这个电路会出问题。问题的表现大约就是,有的时候会回传比较慢,如果速度快了,就会乱码。这个乱码的原因,一是怀疑485芯片,后来我觉得不是485芯片的问题。现在想来就是程序的问题。在程序中收和发没有有效的间隔开来,导致了另一端接收错误,或者说单片机接收错误,然后再回传回来就是错的。之前我整的这个这方面的通讯一直出问题,我感觉也是这个原因。
也就基本上验证了。官方历程中给的例子简单的测试是没有问题的。但是经过再多一点的测试不行。简单的测试就是直接USB转ttl到单片机没有问题。这个时候的解决思路可能就是要把这个程序再搞得复杂一些,比如必须在串口收发的时候,配置dma。虽然我只知道dma这三个字母,但不明白它的概念。但是我想说的就是这些,谢谢
回复

使用道具 举报 送花

  • 打卡等级:以坛为家II
  • 打卡总天数:596
  • 最近打卡:2025-12-20 09:04:18

51

主题

397

回帖

860

积分

高级会员

积分
860
发表于 2025-10-12 21:15:04 | 显示全部楼层
现在我已经基本确定,串口通信做的不好就是因为程序的问题。可是 我还找不到解决的办法。我说的程序的问题说的就是什么也不改的程序。
001.png
回复

使用道具 举报 送花

  • 打卡等级:以坛为家II
  • 打卡总天数:596
  • 最近打卡:2025-12-20 09:04:18

51

主题

397

回帖

860

积分

高级会员

积分
860
发表于 2025-10-12 21:42:46 | 显示全部楼层
可能有人看出来我做了什么。我要说的是。这说明的就是程序有问题。

222.png
回复

使用道具 举报 送花

  • 打卡等级:以坛为家I
  • 打卡总天数:271
  • 最近打卡:2025-12-21 08:44:34

814

主题

1万

回帖

2万

积分

管理员

积分
21407
发表于 2025-10-12 22:12:45 | 显示全部楼层
RS485通信,一般不用 115200bps,
你实际可能 38400bps以下就足够了.


下图的自动切换收发电路,波特率高不了;
你从 9600开始往上测试
截图202510122213188440.jpg




回复

使用道具 举报 送花

  • 打卡等级:以坛为家II
  • 打卡总天数:596
  • 最近打卡:2025-12-20 09:04:18

51

主题

397

回帖

860

积分

高级会员

积分
860
发表于 2025-10-13 08:48:37 | 显示全部楼层
神*** 发表于 2025-10-12 22:12
RS485通信,一般不用 115200bps,
你实际可能 38400bps以下就足够了.

自己发的东西,想看没有看到,应该已经发出去了
我昨天做的那个实验,就是把波特率在今天早上调到了
2000000,就是软件上的最高通讯的时候,也是没有任何乱码
所以我现在确定就不是芯片的问题。不是485芯片的问题
也不是单片机,没有加晶振的问题。用stk 8h2k08u补加晶振照样特别稳。
但是对于串口的收发程序感觉现在还不太懂!感觉以前用的那个都是错的。
回复

使用道具 举报 送花

  • 打卡等级:以坛为家II
  • 打卡总天数:596
  • 最近打卡:2025-12-20 09:04:18

51

主题

397

回帖

860

积分

高级会员

积分
860
发表于 2025-10-18 08:12:50 | 显示全部楼层

[08:10:36.434]收←◆B7 FF
[08:10:36.463]发→◇11 22 33 44 □
[08:10:36.482]收←◆F7
[08:10:36.500]收←◆73
[08:10:36.525]发→◇11 22 33 44 □
[08:10:36.542]收←◆F7
[08:10:36.559]收←◆97 FE
[08:10:36.587]发→◇11 22 33 44 □
[08:10:36.607]收←◆F7 7F 73
[08:10:36.652]发→◇11 22 33 44 □
[08:10:36.673]收←◆F7 7F 73
[08:10:36.713]发→◇11 22 33 44 □
[08:10:36.729]收←◆F7
[08:10:36.747]收←◆37 FE
[08:10:36.777]发→◇11 22 33 44 □
[08:10:36.794]收←◆F7
[08:10:36.811]收←◆37 FE
[08:10:36.839]发→◇11 22 33 44 □
[08:10:36.856]收←◆F7
[08:10:36.873]收←◆B7 FF
[08:10:36.901]发→◇11 22 33 44 □
[08:10:36.923]收←◆B7 FF
[08:10:36.963]发→◇11 22 33 44 □
[08:10:36.979]收←◆F7
[08:10:36.996]收←◆73
[08:10:37.025]发→◇11 22 33 44 □
[08:10:37.044]收←◆F7
[08:10:37.061]收←◆EE
[08:10:37.088]发→◇11 22 33 44 □
[08:10:37.106]收←◆F7
[08:10:37.125]收←◆73
[08:10:37.153]发→◇11 22 33 44 □
这个是按个您发的电路图画的效果不对。用的就是串口1和电脑通信的例子然后改了端口为16 17 别的没有改
还有就是你们说可以把发送口改成腿弯输出。我再试试。
结果就是效果还是这样。没有效果。还是错误
回复

使用道具 举报 送花

  • 打卡等级:以坛为家II
  • 打卡总天数:596
  • 最近打卡:2025-12-20 09:04:18

51

主题

397

回帖

860

积分

高级会员

积分
860
发表于 2025-10-20 22:10:22 | 显示全部楼层
现在已经成功了。用P15控制的方法可以。
直接分享不定长接收代码还有用P15控制
  1. /*---------------------------------------------------------------------*/
  2. /* --- STC MCU Limited ------------------------------------------------*/
  3. /* --- STC 1T Series MCU Demo Programme -------------------------------*/
  4. /* --- Mobile: (86)13922805190 ----------------------------------------*/
  5. /* --- Fax: 86-0513-55012956,55012947,55012969 ------------------------*/
  6. /* --- Tel: 86-0513-55012928,55012929,55012966 ------------------------*/
  7. /* --- Web: www.STCAI.com ---------------------------------------------*/
  8. /* --- Web: www.STCMCUDATA.com  ---------------------------------------*/
  9. /* --- BBS: www.STCAIMCU.com  -----------------------------------------*/
  10. /* --- QQ:  800003751 -------------------------------------------------*/
  11. /* 如果要在程序中使用此代码,请在程序中注明使用了STC的资料及程序        */
  12. /*---------------------------------------------------------------------*/
  13. /*************  功能说明    **************
  14. 本例程基于STC8H8K64U为主控芯片的实验箱9进行编写测试,STC8G、STC8H系列芯片可通用参考.
  15. 串口1全双工中断方式收发通讯程序。
  16. 通过PC向MCU发送数据, MCU收到后通过串口1把收到的数据原样返回.
  17. 用定时器做波特率发生器,建议使用1T模式(除非低波特率用12T),并选择可被波特率整除的时钟频率,以提高精度。
  18. 下载时, 选择时钟 22.1184MHz (用户可自行修改频率).
  19. ******************************************/
  20. #include "stc8h.h"       //包含此头文件后,不需要再包含"reg51.h"头文件
  21. #define MAIN_Fosc       11059200L   //定义主时钟(精确计算115200波特率)
  22. typedef     unsigned char   u8;
  23. typedef     unsigned int    u16;
  24. typedef     unsigned long   u32;
  25. #define     Baudrate1           115200L
  26. #define     UART1_BUF_LENGTH    64
  27. u8  TX1_Cnt;    //发送计数
  28. u8  RX1_Cnt;    //接收计数
  29. bit B_TX1_Busy; //发送忙标志
  30. bit rx_js=0;                         //串口接收结束
  31. unsigned char   xdata RX1_js[UART1_BUF_LENGTH]; //接收数据存放数组
  32. unsigned int    RX1_js_index; //接收索引
  33. unsigned char   xdata RX1_Buffer[UART1_BUF_LENGTH]; //接收数据存放数组
  34. void UART1_config(u8 brt);   // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
  35. void PrintString1(u8 *puts);
  36. void Timer0_Isr(void) interrupt 1
  37. {
  38.        
  39. rx_js=1;
  40.        
  41. }
  42. void Timer0_Init(void)                //5毫秒@11.0592MHz
  43. {
  44.         AUXR |= 0x80;                        //定时器时钟1T模式
  45.         TMOD &= 0xF0;                        //设置定时器模式
  46.         TL0 = 0x00;                                //设置定时初始值
  47.         TH0 = 0x28;                                //设置定时初始值
  48.         TF0 = 0;                                //清除TF0标志
  49.         TR0 = 0;                                //定时器0开始计时
  50.         ET0 = 0;                                //使能定时器0中断
  51. }
  52. //========================================================================
  53. // 函数: void main(void)
  54. // 描述: 主函数。
  55. // 参数: none.
  56. // 返回: none.
  57. // 版本: VER1.0
  58. // 日期: 2014-11-28
  59. // 备注:
  60. //========================================================================
  61. void main(void)
  62. {
  63.     P_SW2 |= 0x80;  //扩展寄存器(XFR)访问使能
  64. Timer0_Init();
  65.     P0M1 = 0x30;   P0M0 = 0x30;   //设置P0.4、P0.5为漏极开路(实验箱加了上拉电阻到3.3V)
  66.     P1M1 = 0x30;   P1M0 = 0x30;   //设置P1.4、P1.5为漏极开路(实验箱加了上拉电阻到3.3V)
  67.     P2M1 = 0x3c;   P2M0 = 0x3c;   //设置P2.2~P2.5为漏极开路(实验箱加了上拉电阻到3.3V)
  68.     P3M1 = 0x50;   P3M0 = 0x50;   //设置P3.4、P3.6为漏极开路(实验箱加了上拉电阻到3.3V)
  69.     P4M1 = 0x3c;   P4M0 = 0x3c;   //设置P4.2~P4.5为漏极开路(实验箱加了上拉电阻到3.3V)
  70.     P5M1 = 0x0c;   P5M0 = 0x0c;   //设置P5.2、P5.3为漏极开路(实验箱加了上拉电阻到3.3V)
  71.     P6M1 = 0xff;   P6M0 = 0xff;   //设置为漏极开路(实验箱加了上拉电阻到3.3V)
  72.     P7M1 = 0x00;   P7M0 = 0x00;   //设置为准双向口
  73.             P1M0 = (P1M0 & ~0x60) | 0x80; P1M1 &= ~0xe0;
  74. P15=0;
  75.     UART1_config(1);    // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
  76.     EA = 1; //允许总中断
  77.     PrintString1("STC8H8K64U UART1 Test Programme!\r\n");  //UART1发送一个字符串
  78.     while (1)
  79.     {
  80.                        
  81.                         /*
  82.                        
  83.         if((TX1_Cnt != RX1_Cnt) && (!B_TX1_Busy))   //收到数据, 发送空闲
  84.         {
  85.                                        
  86.                                        
  87.                                               SBUF = RX1_Cnt;     //把收到的数据远样返回
  88.                                         B_TX1_Busy = 1;
  89.         while(B_TX1_Busy);
  90.                                        
  91.             //SBUF = RX1_Buffer[TX1_Cnt];     //把收到的数据远样返回
  92.             //B_TX1_Busy = 1;
  93.             if(++TX1_Cnt >= UART1_BUF_LENGTH)   TX1_Cnt = 0;
  94.         }
  95.                                
  96.                                 */
  97.                        
  98.                         if (rx_js==1)
  99.                                
  100.                         {
  101.                                         //关闭定时器
  102.                                         TR0 = 0;                                //定时器0开始计时
  103.                                         ET0 = 0;                                //使能定时器0中断
  104.                                
  105.                                 //复位串口
  106.                                 B_TX1_Busy = 0;
  107.                                 TX1_Cnt = 0;
  108.                                 RX1_Cnt = 0;
  109.                                 rx_js=0;
  110.                                 P15=1;
  111.                         SBUF = RX1_Buffer[0];    B_TX1_Busy = 1;    while(B_TX1_Busy);
  112.                         SBUF = RX1_Buffer[1];    B_TX1_Busy = 1;    while(B_TX1_Busy);
  113.                         SBUF = RX1_Buffer[2];    B_TX1_Busy = 1;    while(B_TX1_Busy);
  114.                         SBUF = RX1_Buffer[3];    B_TX1_Busy = 1;    while(B_TX1_Busy);
  115.                         SBUF = RX1_Buffer[4];    B_TX1_Busy = 1;    while(B_TX1_Busy);
  116.                         SBUF = RX1_Buffer[5];    B_TX1_Busy = 1;    while(B_TX1_Busy);
  117.                         SBUF = RX1_Buffer[6];    B_TX1_Busy = 1;    while(B_TX1_Busy);
  118.                         SBUF = RX1_Buffer[7];    B_TX1_Busy = 1;    while(B_TX1_Busy);
  119.                         SBUF = RX1_Buffer[8];    B_TX1_Busy = 1;    while(B_TX1_Busy);
  120.                 P15=0;
  121.                        
  122.                         }
  123.                                
  124.                                
  125.     }
  126. }
  127. //========================================================================
  128. // 函数: void PrintString1(u8 *puts)
  129. // 描述: 串口1发送字符串函数。
  130. // 参数: puts:  字符串指针.
  131. // 返回: none.
  132. // 版本: VER1.0
  133. // 日期: 2014-11-28
  134. // 备注:
  135. //========================================================================
  136. void PrintString1(u8 *puts) //发送一个字符串
  137. {
  138.     for (; *puts != 0;  puts++)     //遇到停止符0结束
  139.     {
  140.         SBUF = *puts;
  141.         B_TX1_Busy = 1;
  142.         while(B_TX1_Busy);
  143.     }
  144. }
  145. //========================================================================
  146. // 函数: SetTimer2Baudraye(u16 dat)
  147. // 描述: 设置Timer2做波特率发生器。
  148. // 参数: dat: Timer2的重装值.
  149. // 返回: none.
  150. // 版本: VER1.0
  151. // 日期: 2014-11-28
  152. // 备注:
  153. //========================================================================
  154. void SetTimer2Baudraye(u16 dat)  // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
  155. {
  156.     AUXR &= ~(1<<4);    //Timer stop
  157.     AUXR &= ~(1<<3);    //Timer2 set As Timer
  158.     AUXR |=  (1<<2);    //Timer2 set as 1T mode
  159.     T2H = dat / 256;
  160.     T2L = dat % 256;
  161.     IE2  &= ~(1<<2);    //禁止中断
  162.     AUXR |=  (1<<4);    //Timer run enable
  163. }
  164. //========================================================================
  165. // 函数: void UART1_config(u8 brt)
  166. // 描述: UART1初始化函数。
  167. // 参数: brt: 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
  168. // 返回: none.
  169. // 版本: VER1.0
  170. // 日期: 2014-11-28
  171. // 备注:
  172. //========================================================================
  173. void UART1_config(u8 brt)    // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
  174. {
  175.     /*********** 波特率使用定时器2 *****************/
  176.     if(brt == 2)
  177.     {
  178.         AUXR |= 0x01;       //S1 BRT Use Timer2;
  179.         SetTimer2Baudraye(65536UL - (MAIN_Fosc / 4) / Baudrate1);
  180.     }
  181.     /*********** 波特率使用定时器1 *****************/
  182.     else
  183.     {
  184.         TR1 = 0;
  185.         AUXR &= ~0x01;      //S1 BRT Use Timer1;
  186.         AUXR |=  (1<<6);    //Timer1 set as 1T mode
  187.         TMOD &= ~(1<<6);    //Timer1 set As Timer
  188.         TMOD &= ~0x30;      //Timer1_16bitAutoReload;
  189.         TH1 = (u8)((65536UL - (MAIN_Fosc / 4) / Baudrate1) / 256);
  190.         TL1 = (u8)((65536UL - (MAIN_Fosc / 4) / Baudrate1) % 256);
  191.         ET1 = 0;    //禁止中断
  192.         INTCLKO &= ~0x02;   //不输出时钟
  193.         TR1  = 1;
  194.     }
  195.     /*************************************************/
  196.     SCON = (SCON & 0x3f) | 0x40;    //UART1模式, 0x00: 同步移位输出, 0x40: 8位数据,可变波特率, 0x80: 9位数据,固定波特率, 0xc0: 9位数据,可变波特率
  197. //  PS  = 1;    //高优先级中断
  198.     ES  = 1;    //允许中断
  199.     REN = 1;    //允许接收
  200.     P_SW1 &= 0x3f;
  201.     P_SW1 |= 0x80;      //UART1 switch to, 0x00: P3.0 P3.1, 0x40: P3.6 P3.7, 0x80: P1.6 P1.7, 0xC0: P4.3 P4.4
  202. //  PCON2 |=  (1<<4);   //内部短路RXD与TXD, 做中继, ENABLE,DISABLE
  203.     B_TX1_Busy = 0;
  204.     TX1_Cnt = 0;
  205.     RX1_Cnt = 0;
  206. }
  207. //========================================================================
  208. // 函数: void UART1_int (void) interrupt UART1_VECTOR
  209. // 描述: UART1中断函数。
  210. // 参数: nine.
  211. // 返回: none.
  212. // 版本: VER1.0
  213. // 日期: 2014-11-28
  214. // 备注:
  215. //========================================================================
  216. void UART1_int (void) interrupt 4
  217. {
  218.     if(RI)
  219.     {
  220.         RI = 0;
  221.         //开始定时器               
  222.         TL0 = 0x00;                                //设置定时初始值
  223.         TH0 = 0x28;                                //设置定时初始值
  224.         TR0 = 1;                                //定时器0开始计时
  225.         ET0 = 1;                                //使能定时器0中断
  226.                        
  227.                        
  228.         RX1_Buffer[RX1_Cnt] = SBUF;
  229.         if(++RX1_Cnt >= UART1_BUF_LENGTH)   RX1_Cnt = 0;    //防溢出
  230.     }
  231.     if(TI)
  232.     {
  233.         TI = 0;
  234.         B_TX1_Busy = 0;
  235.     }
  236. }
复制代码
回复

使用道具 举报 送花

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

本版积分规则

QQ|手机版|深圳国芯人工智能有限公司 ( 粤ICP备2022108929号-2 )

GMT+8, 2025-12-21 08:44 , Processed in 0.112260 second(s), 83 queries .

Powered by Discuz! X3.5

© 2001-2025 Discuz! Team.

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