单片机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); //弹出堆栈
单片机中数的表示及运算
单片机中的数是以二进制表示的,分为有符号数和无符号数两种。
有符号数的表示方法
有符号数是指有“ + (正)”、“ − (负)”符号的数。由于单片机采用二进制数,所以只有“1 ”和“0”两种数字,其中用“ 0 ”表示“ + ”,用“ 1 ”表示“ − ” 。单片机中的数据一般只有8位,一般规定最高位为符号位,因为要用1位表示数的符号,所以只有7位用来表示数值,可以表示− 127~+128。
有符号数的表示方法有 3 种:原码、反码和补码。同一有符号数,用 3 种表示方法得到的数是不同的。下面用3种方法来表示两个有符号数+1011101和− 1011101。
(1)原码
用“1”表示“−”,用“0”表示“+”,其他各数保持不变,采用这种方法表示出来的数称为原码。
+1011101用原码表示是01011101,可写成[01011101] 原。
− 1011101用原码表示是11011101,可写成[11011101] 原。
(2)反码
反码是在原码的基础上求得的。对于正的有符号数,其反码与原码相同;对于负的有符号数,其反码除符号位与原码相同外,其他各位数由原码各位数取反得到。
+1011101用反码表示是01011101,可写成[01011101] 反。
− 1011101用反码表示是10100010,可写成[10100010] 反。
(3)补码
补码是在反码的基础上求得的。对于正的有符号数,其补码与反码、原码相同;对于负的有符号数,其补码除符号位与反码一致外,其他数由反码加1得到。
+1011101用补码表示是01011101,可写成[01011101] 补。
− 1011101用补码表示是10100011,可写成[10100011] 补。
有符号数的运算
用原码表示有符号数简单、直观,但在单片机中,如果采用原码进行减法运算,需要很复杂的硬件电路;如果用补码,可以将减法运算变为加法运算,从而省去减法器而简化硬件电路。
例如:用二进制减法运算和补码加法运算分别计算35 − 21。
① 二进制减法运算:35 − 21=00100011 − 00010101=00001110
② 用补码加法运算。
先将算式转换成补码形式,35 − 21=[+35]+[ − 21]= [00100011] 原 +[10010101] 原 =[00100011] 反+ [11101010] 反 =[00100011] 补 +[11101011] 补。
再对补码进行二进制加法运算:
从上面的运算过程可以看出,补码的符号也参与运算,在8位单片机中,由于数据长度只能有8位,上式结果有9位,第9位会自然丢失,补码加法的运算结果与二进制减法的运算结果是一样的,都是00001110=14。
由此可见,用补码的形式进行运算,可以将减法运算转换为加法运算,运算结果仍是正确的,所以单片机普遍采用补码的形式表示有符号数。
无符号数的表示方法
无符号数因为不用符号位, 8 位全部用来表示数据,所以这种方法可以表示的数据范围是 0 ~ 255 。 8位二进制数的不同表示方式的换算关系见表1-6。
表1-6 8位二进制数的不同表示方式的换算关系
表1-6
续表
表1-6
从表1-6中可以看出,对于同一个二进制数,当采用不同的表示方式时,得到的数值是不同的,特别是大于10000000的有符号数。若想确切知道单片机中的二进制数所对应的十进制数是多少,先要了解该二进制数是有符号数还是无符号数,再换算出该二进制数对应的十进制数。
相关问答
单片机 地址什么0000 H 之类的H是什么意思?在单片机中,地址是指内存中存储的数据的位置。地址通常用十六进制表示,以便更好地利用二进制寻址。例如,0000H表示内存的第一个位置,0001H表示第二个位置,以...
【高手求解 单片机 习题设( 20H )=03H,(22H)=50H,(23H)=5AH,(...[最佳回答]CLRA;A清0CLRC;C清0MOVR2,20H;(20H)->R2MOVR1,#22H;22H->R1LOOP:ADDCA,@R1;(A)+(22H)+(C)...
在 单片机 编程中,keil产生的文件中.C。 H 。HEX。UV2之间是什么关系?.c是C语言源程序,具体的代码.h是C语言头文件,宏定义一些函数、变量.HEX是生成的十六进制文件,烧写文件,最终下载到单片机的。.UV2是工程文件,用kei...
单片机 这里帮我解释下08 H 是什么意思?08H是位地址,是一位,R0是寄存器,,是八位,CLR,JNB是位操作指令,怎么去操作八位,$是当前程序指针,这里是指08H为零继续执行JNB08H,$08H是位地址,是一位,R0是寄...
【#关于 单片机 #概念计算题假定,SP= 60H ,A=30H,B=70H,执行下列...[最佳回答]62H30H70H
单片机 DB01 H ,02H,03H,04H这是什么意思?DB一般只在汇编中才出现,表示后面是一些数据.汇编语言通常用这样的方式保存数据表格.数值后面加一个H,表示是十六进制.DB一般只在汇编中才出现,表示后面是...
123= B= H单片机 ?123是十进制,即一百二十三B是表示用二进制表示,1111011BH表示用16进制表示,7BH不知道怎么换算?可以用windows附件中的计算器123是十进制,即一百二十三B是表示...
pic 单片机 ,用户自己定义的头文件. h 怎么加入到工程里面啊。新手。求解?找到你用的编程软件的include文件夹,在哪里应该有很多的.h文件。将你要用的文件粘到哪里,就可以掉用了。找到你用的编程软件的include文件夹,在哪里应该有很...
单片机 28 h 是什么意思?MOV28H,@R4这是51单片机指令系统中形式,但不是正确有指令,就是说51单片机指令系统中没有这条指令,源操作数只能用R0或R1间址寄存器,所以,正确指令是:M...M...
p0端口对应的特殊功能寄存器叫?在计算机网络中,P0端口通常指的是8051系列微控制器的Port0(P0)端口。对于8051系列微控制器,P0端口是一个具有特殊功能的I/O端口,它具有以下特点:1.可编...