KEIL5 里面实现printf的功能,需要修改一个函数
重写:fputc
包含头文件
在当前.c文件中包含这两个头文件
![](https://i-blog.csdnimg.cn/blog_migrate/541502606bd5b14dbedd6aed11485d7d.png)
重写函数
首先从原理图判断单片机芯片与上位机(电脑)通信是通过那个串口,然后再配置对应的串口
void Usart1_Config(void)
{
GPIO_InitTypeDef GPIO_InitStruct;
USART_InitTypeDef USART_InitStruct;
NVIC_InitTypeDef NVIC_InitStruct;
//1,配置IO口模式
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
GPIO_InitStruct.GPIO_Mode=GPIO_Mode_AF_PP;//复用推挽功能
GPIO_InitStruct.GPIO_Pin=GPIO_Pin_9;//9号引脚
GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;//速度最大
GPIO_Init(GPIOA,&GPIO_InitStruct);
GPIO_InitStruct.GPIO_Mode=GPIO_Mode_IN_FLOATING;//浮空输入
GPIO_InitStruct.GPIO_Pin=GPIO_Pin_10;//10号引脚
GPIO_Init(GPIOA,&GPIO_InitStruct);
//配置串口模式
//1,开串口1时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);
//2,配置串口1的模式
USART_InitStruct.USART_BaudRate = 115200;//串口波特率
USART_InitStruct.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//不适用硬件控制
USART_InitStruct.USART_Mode = USART_Mode_Rx|USART_Mode_Tx;//开启发送和接收
USART_InitStruct.USART_Parity = USART_Parity_No;//不使用奇偶校验
USART_InitStruct.USART_StopBits = USART_StopBits_1;//1位停止位
USART_InitStruct.USART_WordLength = USART_WordLength_8b;//8位字长
USART_Init(USART1,&USART_InitStruct);
//3,使能串口
USART_Cmd(USART1, ENABLE);
//中端配置
//1,开启中断
//开启接收中断
USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);
//开启空闲中断
USART_ITConfig(USART1, USART_IT_IDLE, ENABLE);
//初始化中断
//中断配置
NVIC_InitStruct.NVIC_IRQChannel=USART1_IRQn;//串口的中断通道
NVIC_InitStruct.NVIC_IRQChannelCmd=ENABLE;//使能中断
NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority=0;//抢占优先级
NVIC_InitStruct.NVIC_IRQChannelSubPriority=0;//响应优先级
NVIC_Init(&NVIC_InitStruct);
中断服务函数(标识符命名有点不规范,大家不要学习)
//保存接收的数据
uint8_t recv_Data[256] = {0};
uint8_t recv_Flag=0;
uint8_t recv_Cnt=0;
//中断服务函数 12345678
void USART1_IRQHandler(void)
{
uint8_t data = 0;//变量
//接收数据中断 8次 读取SR的RXNE
if(USART_GetITStatus(USART1, USART_IT_RXNE) == SET)
{
//接收数据--读取USART->DR的数据,会清这个中断
data = USART_ReceiveData(USART1);
recv_Data[recv_Cnt++] = data;//数据先保存到数组内
//清除接收中断 对SR的RXNE位 清0
USART_ClearITPendingBit(USART1, USART_IT_RXNE);
}
//空闲中断 1次 读取SR的IDLE
if(USART_GetITStatus(USART1,USART_IT_IDLE) == SET)
{
recv_Flag = 1;
//中断到了空闲,已经没有数据了,把寄存器内的数据 赋值到某个变量,叫读取
//清除空闲中断
data = USART1->SR;//先读SR,其实上面的USART_GetITStatus函数已经读取过一次,可以不写
data = USART1->DR;//再读DR
}
}
//串口发送字符
void USART1_SendChar(uint8_t p)
{
//能不能在上个字符没有发完的时候,再去发送新的?等待数据发送完成
while(USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
USART_SendData(USART1, p);
}
//串口发送字符串
void USART1_SendStr(uint8_t *p)
{
while(*p != '\0')
{
//能不能在上个字符没有发完的时候,再去发送新的?等待数据发送完成
while(USART_GetFlagStatus(USART1,USART_FLAG_TC) == RESET);
USART_SendData(USART1,*p);
p++;
}
}
//这个函数不用声明,也不要调用
int fputc(int ch, FILE *f)
{
USART1_SendChar(ch);
return ch;
}
打开微库
![](https://i-blog.csdnimg.cn/blog_migrate/b7a6d76f557abd539d5841f15edac2da.png)