日常踩坑记录
51单片机使用printf为什么编译通过了,但是没有效果 ?
MCU型号:STC12C5A60S2
晶振 :11.0592MHZ
实验内容:利用单片机和PC机串口通信的
(晶振为11.0592MHZ)设置波特率为9600
第一个版本的代码如下**(不能正确输出的代码)**:
#include <>
#include ""
#include ""
//----------------------------串口变量------------------------------------//
#define Max_BUFF_Len 50 //数据缓存最大值
unsigned char Rec; //串口数据接收暂存位
unsigned char Rx_Buff[Max_BUFF_Len]; //缓存数组
unsigned int Rx_Count; //数据缓存存放地址
//------------------------------------------------------------------------//
void Uart_init()
{
TMOD|=0X21;
TH1=0Xfd;
TL1=0Xfd; //设置波特率9600
TR1=1; // 开中断1
SM0=0;
SM1=1;
REN=1;
ES=1;
TH0=0xcd; //1ms定时
TL0=0xd4;
ET0=1;
TR0 = 1; //开定时器0
EA=1; //开总中断
}
int main() //主函数
{
Uart_init(); //串口初始化
while(1)
{
printf("Hello Word\r\n"); //输出Hello Word
}
}
void Timer0() interrupt 1
{
TH0=0xcd; //1ms定时
TL0=0xd4;
if(Rx_Buff[Rx_Count-1] == 0x0a || Rx_Count == Max_BUFF_Len) //如果接收到尾标识是换行符(或者等于最大接收数就清空重新接收)
{
if((strstr((char*)Rx_Buff,"OK"))||(strstr((char *)Rx_Buff,">"))) //检测到关键字符
{
Rx_Count=0;memset(Rx_Buff, 0, sizeof(Rx_Buff)); //清空数组,为下一次数据做准备
printf("Data Get OK\r\n");
P1=~P1; //接收成功反转P1端口
}
else
{
Rx_Count=0; //不是需要的数据或者达到最大接收数则开始重新接收
}
}
}
void Usart0() interrupt 4 //串口中断
{
if(RI)
{
RI=0; //接收中断清零
Rec=SBUF; //将接收的数据暂存到Rec
Rx_Buff[Rx_Count]=Rec; //将接收到的数据存入数组
Rx_Count++; //存放地址+1
}
}
经过一番捣鼓,查看51单片机状态控制寄存器SCON寄存器才恍然大悟。
TI 为发送中断标志位,由硬件置位,软件清除。工作方式0中在发送第8位末尾由硬件置位;在其他工作方式时,在发送停止位开始时由硬件置位。TI=1时,申请中断。CPU响应中断后,发送下一帧数据。在任何工作方式中都必须由软件清除TI。:
分析代码后发现,程序运行到while(!TI)停止在该句,因为初始化后TI默认为0,而且还没有发送过数据,TI一直为0,因此程序不会继续向下执行。
解决方法:修改Uart_init()函数,添加TI = 1启动发送。
void Uart_init()
{
TMOD|=0X21;
TH1=0Xfd;
TL1=0Xfd; //设置波特率9600
TR1=1; // 开中断1
SM0=0;
SM1=1;
REN=1;
ES=1;
TH0=0xcd; //1ms定时
TL0=0xd4;
ET0=1;
TR0 = 1; //开定时器0
EA=1; //开总中断
TI=1; //使用printf输出必须打开TI
}
再重新下载到开发板进行调试,完美解决。
又是解决小bug的一天。。。