没有数据乱码,就是分包的问题,加大串口接收软件的超时时间可勉强解决。
根本原因是程序问题,发送字节之间或接收字节之间有不确定的间隔时间,
网上买的USB转串口是采用专用硬件芯片,是实时处理的,发送时字节是连续的,接收的字节也立即上传到电脑端。
你做的这个,程序处理有很多查询操作,中断有延时,还有缓冲区来回复制,没有双缓冲。因此字节流是间断的。
字节之间延时稍微一大,就被判定为分包结束了。
这个是两个买的。同样有问题。所以说这个还是硬件有问题。我想到一个方法不过还没有测试。如果测试成功。这个问题就算是真正解决了。
以前就是因为看到网上买的也是这样的问题。所以就认为有这个问题。其实整好了是没有的。
串口工具上面一般都有超时时间可以设置,两个数据之间间隔时间超过这个时间就会分行显示:
乘风飞扬 发表于 2025-10-9 09:27
串口工具上面一般都有超时时间可以设置,两个数据之间间隔时间超过这个时间就会分行显示:
...
非常非常感谢你的回复。但是事情远远没有这么简单
第一件事就是论响应时间。单片机的响应时间,有时候或者说大多数时候可以搞到一毫秒之内。当然,这样可能要建立在波特率在嗯9600或者115200的时候,或者说115200的时候会更加真实。
我试过用stc的官方程序,然后做了修改之后嗯,会有这个效果,也就是说串口工具发过去会瞬间发回来。但是这是在USB转ttl的操作下实现的。如果是用USB,转485再去到485芯片,再去到stc单片机,目前我用的这个电路会出问题。问题的表现大约就是,有的时候会回传比较慢,如果速度快了,就会乱码。这个乱码的原因,一是怀疑485芯片,后来我觉得不是485芯片的问题。现在想来就是程序的问题。在程序中收和发没有有效的间隔开来,导致了另一端接收错误,或者说单片机接收错误,然后再回传回来就是错的。之前我整的这个这方面的通讯一直出问题,我感觉也是这个原因。
也就基本上验证了。官方历程中给的例子简单的测试是没有问题的。但是经过再多一点的测试不行。简单的测试就是直接USB转ttl到单片机没有问题。这个时候的解决思路可能就是要把这个程序再搞得复杂一些,比如必须在串口收发的时候,配置dma。虽然我只知道dma这三个字母,但不明白它的概念。但是我想说的就是这些,谢谢
现在我已经基本确定,串口通信做的不好就是因为程序的问题。可是 我还找不到解决的办法。我说的程序的问题说的就是什么也不改的程序。
可能有人看出来我做了什么。我要说的是。这说明的就是程序有问题。
RS485通信,一般不用 115200bps,
你实际可能 38400bps以下就足够了.
下图的自动切换收发电路,波特率高不了;
你从 9600开始往上测试
神农鼎 发表于 2025-10-12 22:12
RS485通信,一般不用 115200bps,
你实际可能 38400bps以下就足够了.
自己发的东西,想看没有看到,应该已经发出去了
我昨天做的那个实验,就是把波特率在今天早上调到了
2000000,就是软件上的最高通讯的时候,也是没有任何乱码
所以我现在确定就不是芯片的问题。不是485芯片的问题
也不是单片机,没有加晶振的问题。用stk 8h2k08u补加晶振照样特别稳。
但是对于串口的收发程序感觉现在还不太懂!感觉以前用的那个都是错的。
神农鼎 发表于 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 别的没有改
还有就是你们说可以把发送口改成腿弯输出。我再试试。
结果就是效果还是这样。没有效果。还是错误
现在已经成功了。用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;
}
}