# 3.C语言函数的参数和返回值
## 形式参数和实际参数
前面已经介绍过,函数的参数分为形参和实参两种。在本节中,进一步介绍形参、实参的特点和两者的关系。形参出现在函数定义中,在整个函数体内都可以使用,离开该函数则不能使用。
实参出现在主调函数中,进入被调函数后,实参变量也不能使用。形参和实参的功能是作数据传送。发生函数调用时,主调函数把实参的值传送给被调函数的形参从而实现主调函数向被调函数的数据传送。
函数的形参和实参具有以下特点:
* 形参变量只有在被调用时才分配内存单元,在调用结束时,即刻释放所分配的内存单元。因此,形参只有在函数内部有效。函数调用结束返回主调函数后则不能再使用该形参变量。
* 实参可以是常量、变量、表达式、函数等,无论实参是何种类型的量,在进行函数调用时,它们都必须具有确定的值,以便把这些值传送给形参。因此应预先用赋值,输入等办法使实参获得确定值。
* 实参和形参在数量上,类型上,顺序上应严格一致,否则会发生类型不匹配”的错误。
函数调用中发生的数据传送是单向的。即只能把实参的值传送给形参,而不能把形参的值反向地传送给实参。 因此在函数调用过程中,形参的值发生改变,而实参中的值不会变化。
![](http://www.lvtao.net/content/uploadfile/201404/b7df0e5e3563a10ab62151024fc900e620140421042907.gif)
【例8-2】可以说明这个问题。
~~~
main(){
int n;
printf("input number\n");
scanf("%d",&n);
s(n);
printf("n=%d\n",n);
}
int s(int n){
int i;
for(i=n-1;i>=1;i--)
n=n+i;
printf("n=%d\n",n);
}
~~~
本程序中定义了一个函数s,该函数的功能是求∑ni的值。在主函数中输入n值,并作为实参,在调用时传送给s 函数的形参量n( 注意,本例的形参变量和实参变量的标识符都为n,但这是两个不同的量,各自的作用域不同)。在主函数中用printf 语句输出一次n值,这个n值是实参n的值。在函数s中也用printf 语句输出了一次n值,这个n值是形参最后取得的n值0。从运行情况看,输入n值为100。即实参n的值为100。把此值传给函数s时,形参n的初值也为100,在执行函数过程中,形参n的值变为5050。返回主函数之后,输出实参n的值仍为100。可见实参的值不随形参的变化而变化。
## 函数的返回值
函数的值(或称函数返回值)是指函数被调用之后,执行函数体中的程序段所取得的并返回给主调函数的值。如调用正弦函数取得正弦值,调用【例8-1】的max函数取得的最大数等。对函数的值有以下一些说明:
1) 函数的值只能通过return语句返回主调函数。return语句的一般形式为:
return 表达式;
或者为:
return (表达式);
该语句的功能是计算表达式的值,并返回给主调函数。在函数中允许有多个return语句,但每次调用只能有一个return 语句被执行,因此只能返回一个函数值。
2) 函数值的类型和函数定义中函数的类型应保持一致。如果两者不一致,则以函数类型为准,自动进行类型转换。
3) 如函数值为整型,在函数定义时可以省去类型说明。
4) 不返回函数值的函数,可以明确定义为“空类型”,类型说明符为“void”。如【例8-2】中函数s并不向主函数返函数值,因此可定义为:
~~~
void s(int n){
/* …… */
}
~~~
一旦函数被定义为空类型后,就不能在主调函数中使用被调函数的函数值了。例如,在定义s为空类型后,在主函数中写下述语句
~~~
sum=s(n);
~~~
就是错误的。
为了使程序有良好的可读性并减少出错, 凡不要求返回值的函数都应定义为空类型。
- 前言
- 一. C语言概述
- 1.C语言的发展及其版本
- 2.C语言工作原理和运行机制
- 3.C语言编译器(开发工具|IDE)推荐
- 4.C语言的特点
- 5.第一个C语言程序
- 6.C语言输出函数(printf)和输入函数(scanf)
- 7.C语言程序的结构特点
- 8.C语言字符集
- 9.C语言词汇
- 二. C语言算法
- 1.什么是算法|算法的概念
- 2.简单的C语言算法举例
- 3.C语言算法的特性
- 4.用流程图表示算法
- 5.三种基本结构的流程图
- 6.用N-S流程图表示算法
- 7.用计算机语言表示算法
- 三. 数据类型和运算符
- 1.C语言的数据类型
- 2.C语言常量与变量
- 3.C语言整型数据
- 4.C语言实型数据
- 5.C语言字符型数据
- 6.C语言变量赋初值
- 7.C语言数据类型转换
- 8.C语言算术运算符和算术表达式
- 9.C语言赋值运算符和赋值表达式
- 10.C语言逗号运算符和逗号表达式
- 四. 顺序程序设计
- 1.C语言语句概述
- 2.C语言赋值语句详解
- 3.C语言数据的输入输出
- 4.C语言字符的输入输出
- 7.C语言顺序结构程序设计举例
- 五. 分支结构
- 1.C语言关系运算符和表达式
- 2.C语言逻辑运算符和表达式
- 3.C语言if语句详解
- 4.C语言switch语句的用法详解
- 5.C语言条件运算符和条件表达式
- 6.C语言分支结构程序举例
- 六. 循环控制
- 1.C语言循环控制概述
- 2.C语言goto语句以及用goto语句构成循环
- 3.C语言while语句的用法
- 4.C语言do-while语句的用法
- 5.C语言for语句用法详解
- 6.C语言几种循环的比较
- 7.C语言break和continue语句的用法
- 8.C语言循环控制程序举例
- 七. C语言数组
- 1.C语言一维数组的定义和引用
- 2.C语言二维数组的定义和引用
- 3.C语言字符数组及其应用
- 4.C语言常用字符串处理函数
- 5.C语言数组应用举例
- 6.C语言数组小结
- 八. C语言函数
- 1.C语言函数概述
- 2.C语言函数的定义
- 3.C语言函数的参数和返回值
- 4.C语言函数的调用
- 5.C语言函数的嵌套调用
- 6.C语言函数的递归调用
- 7.C语言数组作为函数参数
- 8.C语言局部变量和全局变量
- 9.C语言变量的存储类别
- 九. 预处理命令
- 1.C语言预处理概述
- 2.C语言无参数宏定义
- 3.C语言带参数宏定义
- 4.C语言文件包含命令
- 5.C语言条件编译详解
- 6.C语言预处理指令总结
- 十. C语言指针
- 1.C语言指针的概念
- 2.C语言指针变量
- 3.C语言指针变量作为函数参数
- 4.C语言指针变量的运算
- 5.C语言数组指针
- 6.C语言通过指针引用数组
- 7.C语言数组名作函数参数
- 8.C语言指向多维数组的指针
- 9.C语言字符串指针
- 10.C语言字符串指针变量与字符数组的区别
- 11.C语言函数指针变量
- 12.C语言指针型函数
- 13.C语言指针数组的概念
- 14.C语言指向指针的指针
- 15.C语言main函数参数
- 16.关于指针的总结
- 十一. 结构体和共用体
- 1.C语言结构体的定义
- 2.C语言结构类型变量的说明
- 3.C语言结构变量成员的表示方法
- 4.C语言结构变量的赋值
- 5.C语言结构变量的初始化
- 6.C语言结构体数组的定义
- 7.C语言指向结构体变量的指针
- 8.C语言指向结构体数组的指针
- 9.C语言结构体指针变量作函数参数
- 10.C语言动态存储分配
- 11.C语言链表的概念
- 12.C语言枚举类型
- 13.C语言类型定义符typedef
- 十二. 位运算
- 1.C语言位运算符详解
- 2.C语言位域(位段)
- 3.关于位运算的总结
- 十三. 文件操作
- 1.C语言文件概述
- 2.C语言文件指针
- 3.C语言文件的打开与关闭
- 4.C语言文件的读写
- 5.C语言文件的随机读写
- 6.C语言文件检测函数
- 7.C语言库文件(头文件)有哪些
- 8.文件操作小结