51单⽚机实现scanf和printf函数
最开始学习C语⾔时,使⽤printf和scanf进⾏格式化输⼊输出⼗分⽅便。
学习单⽚机有很长时间了,之前要再屏幕上显⽰⼀个变量或者通过串⼝传出⼀些变量值观测的话,需要进⾏⼀系列的取余取整运算,很是⿇烦。
最近⼜研究了⼀下keil中针对printf和scanf的实现机理,做了⼀些改动,实现了标准格式化输⼊输出,共⼤家参考。
1.printf函数在格式化输出时,向下调⽤了char putchar(char c);这个函数,在“stdio.h”⾥可以发现有这个函数,所以我们需要⾃⼰构造⼀个这样的函数,即通过串⼝putchar(),代码如下:
char putchar(char c)
{
hal_uart_putchar(c);
return c;
}
其中hal_uart_putchar(c);函数是我们⽐较熟悉的了,是51单⽚机通过串⼝发送⼀个字节的函数,具体代码如下:
void hal_uart_putchar(char i)
{
ES = 0;
TI = 0; //清空发送完中断请求标志位
SBUF = i;  //将数据放⼊寄存器发送
while(TI == 0);//等待发送完毕,发送完毕 TI == 1
复合函数求导怎么求TI = 0; //清空发送完中断请求标志位
ES = 1;
}
有了这两个函数,在单⽚机启动后,⾸先进⾏串⼝初始化,接着就可以使⽤printf了……是不是很简单……
putchar函数
-------------------------------------------------------------------------------------------------------------------------------------
2.下⾯再看scanf的具体实现⽅法:
scanf函数在格式化输⼊时,向下掉⽤了char getkey(void);这个函数,在“stdio.h”⾥可以发现有这个函数,所以我们需要⾃⼰构造⼀个这样的函数,即通过串⼝getkey(),代码如下:
char _getkey (void)
{
return hal_uart_getchar();
}
其中hal_uart_getchar();稍稍复杂,但也很好理解,代码如下:
char hal_uart_getchar(void)
{
uchar ch;
//Wait until a character is available:
while(uart_rx_cnt == 0);
ES = 0;
ch = uart_rx[uart_rx_rp];
error: access denieduart_rx_rp = (uart_rx_rp + 1) % UART_BUF_SIZE;
uart_rx_cnt--;
ES = 1;
return ch;
}
这个函数是从串⼝接收队列中取出队尾的⼀个字节。uart_rx_cnt 表⽰现在串⼝队列中的已有字节数,uart_rx_rp指向队尾。
最后要介绍的⼀个函数是串⼝接收中断函数,代码如下:
void UART1InterruptReceive(void) interrupt 4
{
ES=0;//关串⾏⼝中断
if(RI)
{
excel函数上下左右的公式RI=0;//接收中断信号清零,表⽰将继续接收
if(uart_rx_cnt < UART_BUF_SIZE)
{
uart_rx[uart_rx_wp] = SBUF;
uart_rx_wp = (uart_rx_wp + 1) % UART_BUF_SIZE;
uart_rx_cnt++;
}
}
ES=1;//开串⾏⼝中断
}
该函数实现了串⼝的中断接收,收到的新的字节存放在队⾸,即uart_rx_wp指向队列的⾸地址,每次收到⼀个新的字节,uart_rx_cnt增1。
英语名字shelly
⾄此,scanf函数也可以实现了。activex下载
测试截图:
注:串⼝接收的队列没有溢出检测……
这篇⽂章⾥实现的是对于串⼝的格式化输⼊输出,实际上,我们同样可以对hal_uart_getchar();和hal_uart_putchar(c);函数进⾏更改,实现在屏幕上的格式化输出等,思路都是⼀样的……
有不合理的地⽅,请⼤家批评指正。

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。