health 发表于 2025-10-8 19:03:12

没有数据乱码,就是分包的问题,加大串口接收软件的超时时间可勉强解决。

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

qepuemc 发表于 2025-10-9 08:04:25



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

乘风飞扬 发表于 2025-10-9 09:27:08



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

qepuemc 发表于 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这三个字母,但不明白它的概念。但是我想说的就是这些,谢谢

qepuemc 发表于 2025-10-12 21:15:04

现在我已经基本确定,串口通信做的不好就是因为程序的问题。可是 我还找不到解决的办法。我说的程序的问题说的就是什么也不改的程序。

qepuemc 发表于 2025-10-12 21:42:46

可能有人看出来我做了什么。我要说的是。这说明的就是程序有问题。


神农鼎 发表于 2025-10-12 22:12:45

RS485通信,一般不用 115200bps,
你实际可能 38400bps以下就足够了.


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





qepuemc 发表于 2025-10-13 08:48:37

神农鼎 发表于 2025-10-12 22:12
RS485通信,一般不用 115200bps,
你实际可能 38400bps以下就足够了.



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

qepuemc 发表于 2025-10-18 08:12:50

神农鼎 发表于 2025-10-7 13:46

收←◆B7 FF
发→◇11 22 33 44 □
收←◆F7
收←◆73
发→◇11 22 33 44 □
收←◆F7
收←◆97 FE
发→◇11 22 33 44 □
收←◆F7 7F 73
发→◇11 22 33 44 □
收←◆F7 7F 73
发→◇11 22 33 44 □
收←◆F7
收←◆37 FE
发→◇11 22 33 44 □
收←◆F7
收←◆37 FE
发→◇11 22 33 44 □
收←◆F7
收←◆B7 FF
发→◇11 22 33 44 □
收←◆B7 FF
发→◇11 22 33 44 □
收←◆F7
收←◆73
发→◇11 22 33 44 □
收←◆F7
收←◆EE
发→◇11 22 33 44 □
收←◆F7
收←◆73
发→◇11 22 33 44 □
这个是按个您发的电路图画的效果不对。用的就是串口1和电脑通信的例子然后改了端口为16 17 别的没有改
还有就是你们说可以把发送口改成腿弯输出。我再试试。
结果就是效果还是这样。没有效果。还是错误

qepuemc 发表于 2025-10-20 22:10:22

现在已经成功了。用P15控制的方法可以。
直接分享不定长接收代码还有用P15控制
/*---------------------------------------------------------------------*/
/* --- STC MCU Limited ------------------------------------------------*/
/* --- STC 1T Series MCU Demo Programme -------------------------------*/
/* --- Mobile: (86)13922805190 ----------------------------------------*/
/* --- Fax: 86-0513-55012956,55012947,55012969 ------------------------*/
/* --- Tel: 86-0513-55012928,55012929,55012966 ------------------------*/
/* --- Web: www.STCAI.com ---------------------------------------------*/
/* --- Web: www.STCMCUDATA.com---------------------------------------*/
/* --- BBS: www.STCAIMCU.com-----------------------------------------*/
/* --- QQ:800003751 -------------------------------------------------*/
/* 如果要在程序中使用此代码,请在程序中注明使用了STC的资料及程序      */
/*---------------------------------------------------------------------*/

/*************功能说明    **************

本例程基于STC8H8K64U为主控芯片的实验箱9进行编写测试,STC8G、STC8H系列芯片可通用参考.

串口1全双工中断方式收发通讯程序。

通过PC向MCU发送数据, MCU收到后通过串口1把收到的数据原样返回.

用定时器做波特率发生器,建议使用1T模式(除非低波特率用12T),并选择可被波特率整除的时钟频率,以提高精度。

下载时, 选择时钟 22.1184MHz (用户可自行修改频率).

******************************************/

#include "stc8h.h"       //包含此头文件后,不需要再包含"reg51.h"头文件

#define MAIN_Fosc       11059200L   //定义主时钟(精确计算115200波特率)


typedef   unsigned char   u8;
typedef   unsigned int    u16;
typedef   unsigned long   u32;


#define   Baudrate1         115200L
#define   UART1_BUF_LENGTH    64


u8TX1_Cnt;    //发送计数
u8RX1_Cnt;    //接收计数
bit B_TX1_Busy; //发送忙标志

bit rx_js=0;                         //串口接收结束

unsigned char   xdata RX1_js; //接收数据存放数组

unsigned int    RX1_js_index; //接收索引
unsigned char   xdata RX1_Buffer; //接收数据存放数组

void UART1_config(u8 brt);   // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
void PrintString1(u8 *puts);









void Timer0_Isr(void) interrupt 1
{
       
rx_js=1;
       
}

void Timer0_Init(void)                //5毫秒@11.0592MHz
{
        AUXR |= 0x80;                        //定时器时钟1T模式
        TMOD &= 0xF0;                        //设置定时器模式
        TL0 = 0x00;                                //设置定时初始值
        TH0 = 0x28;                                //设置定时初始值
        TF0 = 0;                                //清除TF0标志
        TR0 = 0;                                //定时器0开始计时
        ET0 = 0;                                //使能定时器0中断
}







//========================================================================
// 函数: void main(void)
// 描述: 主函数。
// 参数: none.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void main(void)
{
    P_SW2 |= 0x80;//扩展寄存器(XFR)访问使能
Timer0_Init();
    P0M1 = 0x30;   P0M0 = 0x30;   //设置P0.4、P0.5为漏极开路(实验箱加了上拉电阻到3.3V)
    P1M1 = 0x30;   P1M0 = 0x30;   //设置P1.4、P1.5为漏极开路(实验箱加了上拉电阻到3.3V)
    P2M1 = 0x3c;   P2M0 = 0x3c;   //设置P2.2~P2.5为漏极开路(实验箱加了上拉电阻到3.3V)
    P3M1 = 0x50;   P3M0 = 0x50;   //设置P3.4、P3.6为漏极开路(实验箱加了上拉电阻到3.3V)
    P4M1 = 0x3c;   P4M0 = 0x3c;   //设置P4.2~P4.5为漏极开路(实验箱加了上拉电阻到3.3V)
    P5M1 = 0x0c;   P5M0 = 0x0c;   //设置P5.2、P5.3为漏极开路(实验箱加了上拉电阻到3.3V)
    P6M1 = 0xff;   P6M0 = 0xff;   //设置为漏极开路(实验箱加了上拉电阻到3.3V)
    P7M1 = 0x00;   P7M0 = 0x00;   //设置为准双向口
          P1M0 = (P1M0 & ~0x60) | 0x80; P1M1 &= ~0xe0;
P15=0;

    UART1_config(1);    // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
    EA = 1; //允许总中断

    PrintString1("STC8H8K64U UART1 Test Programme!\r\n");//UART1发送一个字符串

    while (1)
    {
                       
                        /*
                       
      if((TX1_Cnt != RX1_Cnt) && (!B_TX1_Busy))   //收到数据, 发送空闲
      {
                                       
                                       
                                              SBUF = RX1_Cnt;   //把收到的数据远样返回
                                        B_TX1_Busy = 1;
      while(B_TX1_Busy);
                                       
            //SBUF = RX1_Buffer;   //把收到的数据远样返回
            //B_TX1_Busy = 1;
            if(++TX1_Cnt >= UART1_BUF_LENGTH)   TX1_Cnt = 0;

      }
                               
                                */
                       
                        if (rx_js==1)
                               
                        {
                                        //关闭定时器
                                        TR0 = 0;                                //定时器0开始计时
                                        ET0 = 0;                                //使能定时器0中断
                               
                                //复位串口
                                B_TX1_Busy = 0;
                                TX1_Cnt = 0;
                                RX1_Cnt = 0;
                                rx_js=0;
                                P15=1;
                        SBUF = RX1_Buffer;    B_TX1_Busy = 1;    while(B_TX1_Busy);
                        SBUF = RX1_Buffer;    B_TX1_Busy = 1;    while(B_TX1_Busy);
                        SBUF = RX1_Buffer;    B_TX1_Busy = 1;    while(B_TX1_Busy);
                        SBUF = RX1_Buffer;    B_TX1_Busy = 1;    while(B_TX1_Busy);
                        SBUF = RX1_Buffer;    B_TX1_Busy = 1;    while(B_TX1_Busy);
                        SBUF = RX1_Buffer;    B_TX1_Busy = 1;    while(B_TX1_Busy);
                        SBUF = RX1_Buffer;    B_TX1_Busy = 1;    while(B_TX1_Busy);
                        SBUF = RX1_Buffer;    B_TX1_Busy = 1;    while(B_TX1_Busy);
                        SBUF = RX1_Buffer;    B_TX1_Busy = 1;    while(B_TX1_Busy);
                P15=0;
                       
                        }
                               
                               
    }
}


//========================================================================
// 函数: void PrintString1(u8 *puts)
// 描述: 串口1发送字符串函数。
// 参数: puts:字符串指针.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void PrintString1(u8 *puts) //发送一个字符串
{
    for (; *puts != 0;puts++)   //遇到停止符0结束
    {
      SBUF = *puts;
      B_TX1_Busy = 1;
      while(B_TX1_Busy);
    }
}

//========================================================================
// 函数: SetTimer2Baudraye(u16 dat)
// 描述: 设置Timer2做波特率发生器。
// 参数: dat: Timer2的重装值.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void SetTimer2Baudraye(u16 dat)// 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
{
    AUXR &= ~(1<<4);    //Timer stop
    AUXR &= ~(1<<3);    //Timer2 set As Timer
    AUXR |=(1<<2);    //Timer2 set as 1T mode
    T2H = dat / 256;
    T2L = dat % 256;
    IE2&= ~(1<<2);    //禁止中断
    AUXR |=(1<<4);    //Timer run enable
}

//========================================================================
// 函数: void UART1_config(u8 brt)
// 描述: UART1初始化函数。
// 参数: brt: 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void UART1_config(u8 brt)    // 选择波特率, 2: 使用Timer2做波特率, 其它值: 使用Timer1做波特率.
{
    /*********** 波特率使用定时器2 *****************/
    if(brt == 2)
    {
      AUXR |= 0x01;       //S1 BRT Use Timer2;
      SetTimer2Baudraye(65536UL - (MAIN_Fosc / 4) / Baudrate1);
    }

    /*********** 波特率使用定时器1 *****************/
    else
    {
      TR1 = 0;
      AUXR &= ~0x01;      //S1 BRT Use Timer1;
      AUXR |=(1<<6);    //Timer1 set as 1T mode
      TMOD &= ~(1<<6);    //Timer1 set As Timer
      TMOD &= ~0x30;      //Timer1_16bitAutoReload;
      TH1 = (u8)((65536UL - (MAIN_Fosc / 4) / Baudrate1) / 256);
      TL1 = (u8)((65536UL - (MAIN_Fosc / 4) / Baudrate1) % 256);
      ET1 = 0;    //禁止中断
      INTCLKO &= ~0x02;   //不输出时钟
      TR1= 1;
    }
    /*************************************************/

    SCON = (SCON & 0x3f) | 0x40;    //UART1模式, 0x00: 同步移位输出, 0x40: 8位数据,可变波特率, 0x80: 9位数据,固定波特率, 0xc0: 9位数据,可变波特率
//PS= 1;    //高优先级中断
    ES= 1;    //允许中断
    REN = 1;    //允许接收
    P_SW1 &= 0x3f;
    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
//PCON2 |=(1<<4);   //内部短路RXD与TXD, 做中继, ENABLE,DISABLE

    B_TX1_Busy = 0;
    TX1_Cnt = 0;
    RX1_Cnt = 0;
}


//========================================================================
// 函数: void UART1_int (void) interrupt UART1_VECTOR
// 描述: UART1中断函数。
// 参数: nine.
// 返回: none.
// 版本: VER1.0
// 日期: 2014-11-28
// 备注:
//========================================================================
void UART1_int (void) interrupt 4
{
    if(RI)
    {
      RI = 0;
        //开始定时器               
        TL0 = 0x00;                                //设置定时初始值
        TH0 = 0x28;                                //设置定时初始值
        TR0 = 1;                                //定时器0开始计时
        ET0 = 1;                                //使能定时器0中断
                       
                       
      RX1_Buffer = SBUF;
      if(++RX1_Cnt >= UART1_BUF_LENGTH)   RX1_Cnt = 0;    //防溢出
    }

    if(TI)
    {
      TI = 0;
      B_TX1_Busy = 0;
    }
}

页: 1 [2] 3
查看完整版本: 485通信错误用STC单片机和485芯片做USB转串口有错误