51单片机实现scanf和printf函数
我们在使用51单片机时,会特别平凡的用到串口这个外设,但是它没有提供像标准c一样提供scanf和printf这两个输入输出函数,这样平时每次使用起来比较麻烦。但是我们可以自己通过单片机发送一个字节和接收一个字节这两个函数来改造一下,下面请看具体过程。
1.printf函数
在格式化输出时,向下调用了char putchar(char c);这个函数,在“stdio.h”里可以发现有这个函数,所以我们需要自己构造一个这样的函数,即通过串口putchar(),代码如下:
char putchar(char c)
{
al_uart_putchar(c);
hreturn c; }
其中hal_uart_putchar(c);函数是我们比较熟悉的了,是51单片机 通过串口发送一个字节的函数,具体代码如下:
void hal_uart_putchar(char i)
{ ES = 0;
清空发送完中断请求标志位 SBUF = i;
TI = 0; /
/ //将数据放入寄存器发送
//等待发送完毕,发送完毕 TI == 1 TI = 0; //清空发送
while(TI == 0)
;完中断请求标志位 ES = 1; }
有了这两个函数,在单片机启动后,首先进行串口初始化,接着就可以使用printf了……是不是很简单……
-------------------------------------------------------------------------------------------------------------------------------------
2.下面再看scanf的具体实现方法:
scanf函数在格式化输入时,向下掉用了char getkey(void);这个函数,在“stdio.h”里可以发现有这个函数,所以我们需要自己构造一个这样的函数,即通过串口getkey(),代码如下:
char _getkey (void)
{
eturn hal_uart_getchar(); }
r
其中hal_uart_getchar(); 稍稍复杂,但也很好理解,代码如下:
char hal_uart_getchar(void)
{ uchar ch;
a character is available: while(uart_
//Wait unti
lrx_cnt == 0); ES = 0; ch = uart_rx[uart_rx_rp];
rt_rx_cnt--; ES = 1; return ch; }
uart_rx_rp = (uart_rx_rp + 1) % UART_BUF_SIZE; u
a
这个函数是从串口接收队列中取出队尾的一个字节。uart_rx_cnt 表示现在串口队列中的已有字节数,uart_rx_rp 指向队尾。
最后要介绍的一个函数是串口接收中断函数,代码如下:
void UART1InterruptReceive(void) interrupt 4
{ ES=0;//关串行口中断 if(RI) {
if(uart_rx_cnt < UART_BUF
RI=0;//接收中断信号清零,表示将继续接收
_SIZE) { uart_rx[uart_rx_wp] = SBUF;
_SIZE; uart_rx_cnt++; } } ES=1;//开串行口中断
uart_rx_wp = (uart_rx_wp + 1) % UART_BU
F}
该函数实现了串口的中断接收,收到的新的字节存放在队首,即uart_rx_wp指向队列的首地址,每次收到一个新的字节,uart_rx_cnt增1。
至此,scanf函数也可以实现了。
注:串口接收的队列没有溢出检测……
这篇文章里实现的是对于串口的格式化输入输出,实际上,我们同样可以对hal_uart_getchar();和hal_uart_putchar(c);函数进行更改,实现在屏幕上的格式化输出等,思路都是一样的……
欢迎大家评论留言,讨论
单片机C语言之函数
函数定义
函数是一个自我包含的完成一定相关功能的执行代码段。通常C语言的编译器会自带标准的函数库,这些都是一些常用的函数。标准函数已由编译器软件商编写定义,使用者直接调用就能了,而无需定义。但是标准的函数不足以满足使用者的特殊要求,因此C语言允许使用者根据需要编写特定功能的函数,要调用它必须要先对其进行定义。
定义的模式如下:
函数类型 函数名称(形式参数表)
函数类型是说明所定义函数返回值的类型。返回值其实就是一个变量,只要按变量类型来定义函数类型就行了。如函数不需要返回值函数类型能写作“void”表示该函数没有返回值。注意的是函数体返回值的类型一定要和函数类型一致,不然会造成错误。
函数名称的定义在遵循C语言变量命名规则的同时,不能在同一程序中定义同名的函数,这将会造成编译错误(同一程序中是允许有同名变量的,因为变量有全局和局部变量之分)。
形式参数是指调用函数时要传入到函数体内参与运算的变量,它可以是一个、几个或没有。当函数不需要形式参数时(即无参函数),括号内为空或写入“void”表示,但括号不能少。
函数体中能包含有局部变量的定义和程序语句,如函数要返回运算值则要使用return语句进行返回。若在函数的{}号中也能什么也不写,这就成了空函数。在一个程序项目中可以写一些空函数,在以后的修改和升级中能方便的在这些空函数中进行功能扩充。
函数的调用
(一)函数调用的一般说明
函数定义好以后,要被其它函数调用了才能被执行。C语言的函数是能相互调用的,但在调用函数前,必须对函数的类型进行说明,就算是标准库函数也不例外。
标准库函数的说明会被按功能分别写在不一样的头文件中,使用时只要在文件最前面用#include预处理语句引入相应的头文件。如前面使用的printf函数的说明是放在文件名为stdio.h的头文件中。
调用就是指一个函数体中引用另一个已定义的函数来实现所需要的功能,这个时候函数体称为主调用函数,函数体中所引用的函数称为被调用函数。主函数只是相对于被调用函数而言。
一个函数体中能调用数个其它的函数,这些被调用的函数同样也能调用其它函数,也能嵌套调用。但是在c51语言中有一个函数是不能被其它函数所调用的,它就是main主函数。
标准库函数只要用#include引入已写好说明的头文件,在程序就能直接调用函数了。如调用的是自定义的函数则要用如下形式编写函数类型说明:
类型标识符 函数的名称(形式参数表);
这样的说明方式是用在被调函数定义和主调函数是在同一文件中。也能把这些写到文件名.h的文件中用#include“文件名.h”引入。
如果被调函数的定义和主调函数不是在同一文件中的,则要用如下的方式进行说明,说明被调函数的定义在同一项目的不一样文件之上,这样说明的函数也能称为外部函数,定义如下:
extern类型标识符 函数的名称(形式参数表);
函数的定义和说明是完全不一样的,在编译的角度上看函数的定义是把函数编译存放在ROM的某一段地址上,而函数说明是告诉编译器要在程序中使用那些函数并确定函数的地址。如果在同一文件中被调函数的定义在主调函数之前,这个时候能不用说明函数类型。也就是说在main函数之前定义的函数,在程序中就能不用写函数类型说明了。能在一个函数体调用另一个函数(嵌套调用),但不允许在一个函数定义中定义另一个函数。还要注意的是函数定义和说明中的“类型、形参表、名称”等都要相一致。
(二)函数调用的一般形式
调用函数的一般形式如下:
函数名 (实际参数表)
“函数名”就是指被调用的函数。
实际参数表能为零或多个参数,多个参数时要用逗号隔开,每个参数的类型、位置应与函数定义时所的形式参数一一对应,它的作用就是把参数传到被调用函数中的形式参数,如果类型不对应就会产生一些错误。调用的函数是无参函数时不写参数,但不能省后面的括号。
下面我们看一下在实际应用中函数不同的调用方式:
1. 函数语句
例如printf(“Hello World!\n”);
它以“Hello World!\n”为参数调用printf这个库函数,在这里函数调用被看作了一条语句。
2. 函数参数
“函数参数”这种方式是指被调用函数的返回值当作另一个被调用函数的实际参数,如temp=StrToInt(CharB(16));CharB的返回值作为StrToInt函数的实际参数传递。
3. 函数表达式
例如temp=Count();
这个函数的调用作为一个运算对象出现在表达式中,称为函数表达式。例子中Count()返回一个int类型的返回值直接赋值给temp。注意的是这种调用方式要求被调用的函数能返回一个同类型的值,不然会出现不可预料的错误。
C51常用头文件
下面介绍一些常用的C51头文件:
absacc.h——包含允许直接访问8051不同存储区的宏定义;
assert.h——文件定义assert 宏,可以用来建立程序的测试条件;
ctype——字符转换和分类程序;
intrins.h——文件包含指示编译器产生嵌入式固有代码的程序的原型;
math.h——数学程序;
reg51.h——51的特殊寄存器;
reg52.h——52的特殊寄存器;
setjmp.h——定义jmp_buf类型和setjmp和longjmp程序的原型;
stdarg.h——可变长度参数列表程序;
stdlib.h——存储区分配程序;
stdio.h——标准输入和输出程序;
string.h——字符串操作程序、缓冲区操作程序。
对于常用的MCS-51单片机,必须包含reg51.h的头文件,因为该文件对51单片机的相关寄存器及位进行了定义,这样在程序中才可以使用这些资源。
reg51.h文件的具体内容如下:
#ifndef __REG51_H__
#define __REG51_H__
/* BYTE Register */ //单元定义
sfr P0 = 0x80;
sfr P1 = 0x90;
sfr P2 = 0xA 0 ;
sfr P3 = 0xB0;
sfr PSW = 0xD0;
sfr ACC = 0xE0;
sfr B = 0xF0;
sfr SP = 0x81;
sfr DPL = 0x82;
sfr DPH = 0x83;
sfr PCON = 0x87;
sfr TCON = 0x88;
sfr TMOD = 0x89;
sfr TL0 = 0x8A;
sfr TL1 = 0x8B;
sfr TH0 = 0x8C;
sfr TH1 = 0x8D;
sfr IE = 0xA8;
sfr IP = 0xB8;
sfr SCON = 0x98;
sfr SBUF = 0x99;
/* BIT Register */
/* PSW */ //位定义
sbit CY = 0xD7;
sbit AC = 0xD6;
sbit F0 = 0xD5;
sbit RS1 = 0xD4;
sbit RS0 = 0xD3;
sbit OV = 0xD2;
sbit P = 0xD0;
/* TCON */
sbit TF1 = 0x8F;
sbit TR1 = 0x8E;
sbit TF0 = 0x8D;
sbit TR0 = 0x8C;
sbit IE1 = 0x8B;
sbit IT1 = 0x8A;
sbit IE0 = 0x89;
sbit IT0 = 0x88;
/* IE */
sbit EA = 0xAF;
sbit ES = 0xAC;
sbit ET1 = 0xAB;
sbit EX1 = 0xAA;
sbit ET0 = 0xA9;
sbit EX0 = 0xA8;
/* IP */
sbit PS = 0xBC;
sbit PT1 = 0xBB;
sbit PX1 = 0xBA;
sbit PT0 = 0xB9;
sbit PX0 = 0xB8;
/* P3 */
sbit RD = 0xB7;
sbit WR = 0xB6;
sbit T1 = 0xB5;
sbit T0 = 0xB4;
sbit INT1 = 0xB3;
sbit INT0 = 0xB2;
sbit TXD = 0xB1;
sbit RXD = 0xB0;
/* SCON */
sbit SM0 = 0x9F;
sbit SM1 = 0x9E;
sbit SM2 = 0x9D;
sbit REN = 0x9C;
sbit TB8 = 0x9B;
sbit RB8 = 0x9A;
sbit TI = 0x99;
sbit RI = 0x98;
#endif
而absacc.h为对8051单片机的不同存储区的宏定义,具体如下:
#define CBYTE ((unsigned char volatile code *) 0) //定义程序存储器;
#define DBYTE ((unsigned char volatile data *) 0) //定义片内数据存储区;
#define PBYTE ((unsigned char volatile pdata *) 0) //定义页寻址空间;
#define XBYTE ((unsigned char volatile xdata *) 0) //定义片外数据存储区。
而intrins.h文件对指示编译器产生嵌入式代码,如空操作执行、位指令、栈操作指令等,该文件的具体内容如下:
extern void _nop_ (void); //空操作8051 NOP指令
extern bit _testbit_ (bit); //测试并清零位8051 JBC指令
extern unsigned char _cror_ (unsigned char,unsigned char); //字符循环左移
extern unsigned int _iror_ (unsigned int,unsigned char); //字符循环右移
extern unsigned long _lror_ (unsigned long,unsigned char); //整数循环右移
extern unsigned char _crol_ (unsigned char,unsigned char); //整数循环右移
extern unsigned int _irol_ (unsigned int,unsigned char); //整数循环左移
extern unsigned long _lrol_ (unsigned long,unsigned char); //长整数循环左移
extern unsigned char _chkfloat_(float); //测试并返回源点数状态
extern void _push_ (unsigned char _sfr); //压入堆栈
extern void _pop_ (unsigned char _sfr); //弹出堆栈
相关问答
关于C 51单片机 的小数点四舍五入处理?将你需保留的那一位的低一位加5之后舍掉即可单片机内一般采用整数计算,如果要保留两位小数,需将原数据扩大100倍,小数点该向左移2位18754/100=187·54但单片机...
单片机 程序怎么在电脑上调试?用什么软件好?单片机可以实现在线调试程序,通过软件设置断点,进行单步调试,这样可以看到单片机是如何一步步执行程序的,从而快速找到问题,解决问题。以Keil和IAR为例介绍...1,...
在C语言中用\a表示鸣铃是什么意思啊?你编个简单的小程序用输出函数(如printf(''\a'');)输出电脑会发出蜂鸣声printf(“\1”)输出笑脸,可以试试还有好多图形声音和单片机程序相似例如:#...
编辑好C语言程序,怎么导入 单片机 -ZOL问答这个文件就可以用烧录软件下载到单片机ROM。在DOS里有个EDIT编辑器直接输入...printf("-%s\n-----------------------------\n&qu...
C语言中的 -是什么意思 - Annette1907 的回答 - 懂得修饰符格式说明、意义M%md以宽度m输出整型数,不足m时,左补空格0m%0md以宽度m输出整型数,不足m时,左补零m,n%m.nf以宽度m输出实型小数,小数位...
C语言while循环里使用scanf()获取键盘输入,执行循环。scanf()...printf("%d\n",sum);}用dowhile就是先执行后判断,那个下面的也是1加到100的...还有待考虑,这个似乎和cpu有关吧,我只知道单片机中的定时9条...
C语言编程中,输入两个整数?具体输入如下:#include"stdio.h"main(){inta,b;printf("请输入两个整数(a,b):");scanf("%d,...
c语言中怎么调用自己定义的函数? - ?淳某人 的回答 - 懂得在使用一个函数之前必须先对他进行声明://voidB();声明B函数的存在。voidA(){B();//非法,程序执行到此时并不知道B函数的存在。}voidB(){}或者#i...
如何开始STM8S系列 单片机 的开发?这跟单片机没关系,这是C语言的库函数使用时只需将putc函数(被printf函数调用)换成串囗发送函数即可,换了之后优先调用用户编写的putc函数这跟单片机没关系,...
得到C语言源代码后怎么样使用?-ZOL问答甚至包含一些嵌入式处理器(单片机或称MCU)以及超级电脑等作业平台。首先温度也...当然了,mfc是窗口应用程序,printf是命令行那种窗口输出的。安卓用java语言编...