1
第八章:模块化程序设计
重点与要求:
?重点掌握函数的定义、
声明及传值与调用。熟悉函数
的嵌套与递归调用。
?掌握变量的作用域与存储类别。
2
函数
一、函数
C语言中,程序是由函数来实现的。函数可分为( 1):
标准库函数( 2)用户自定义函数。
在 C语言中,至少要有一个 main ( )函数,程序的执行是
从 main( )函数开始的。函数的调用过程如图:
f11( )
{…}
f21( )
{…}
f31( )
{…}
f1( )
{…
f11( );
…}
f2( )
{…
f11( );
f22( );
…}
main( )
{…
f1( );

f2( );
…}
3
函数
其中,每个方框相当于一个模块,而每个函数相当于一
个输入输出系统。 in out
例如:要设计一个如下图所示的管理程序:
高等院校事务管理





























4
函数
由此可以得出:模块化程序设计有如下特点:
( 1)模块相对独立,功能单一,可混合编写也可独立编写
调试。
( 2)可集体开发,缩短开发周期。
( 3)开发出的模块,可在不同的应用程序中多次使用,减
少重复劳动,提高开发效率。
( 4)测试、更新以模块为单位进行而不会影响其他模块,
一、函数的 定义 与 声明
1、定义,[存储类型 ] [数据类型 ] 函数名(带类型的形参名
表)
{ 内部变量说明;
语句;
}
5
函数
例,计算表达式的值
x 2- x+1 (x<0)
x3+x+3 (x>0)
float y (float x )
{ if (x<0)
return (x*x- x+1);
else
return (x*x*x+x+3);
}
注意:当有多个 return语句时,每个 return语句后面
的表达式的类型应相同。
y={
6
函数
如,func (int n)
{ if (n>10)
return (2*x+3);
else
return;
} 编译时,由于第二个 return语句而给出警告。
说明,1)存储类型(后面讲)
2)数据类型:指函数返回值的类型。 若缺省函数
类型,一律按整形处理。
3)函数是平等的兄弟般的关系,它不能嵌套定义
即不能在一个函数体内再定义另一个函数。
4)只有自定义函数而没有 main()函数的程序是没
有意义的。(例 L4—2)
7
函数
2、函数的 声明
[数据类型 ] 函数名 (类型标记符 [形参 ], … );
注意后面的, ;, 不要丢了。它与定义不同,一个函数一般要经过声明
才能使用(就好象变量一样必须先声明才能使用),除非它在调用它
的前面定义。 只有当返回的数据类型为 int,数据类型才可省略。
如,main( )
{ …
double new_style ( int,double ); /*函数声明 */

}
Double new_style (int a,double x) /*函数定义 */
{

}
8
函数
二,return 语句
return 返回值表达式;
return后面表达式的类型必须和函数定义时函数名
前的类型保持一致。
三、函数的调用
函数名 ( [实参表 ]);( 如果是有返回值的函数,
则可将调用结果赋值给别的变量,若是无值函数,
则不能赋值,只能单独调用;(例 4—1打印若干三
角形)
五、形参、实参、参数的传递
形参:定义函数的参数。
实参:调用函数时有具体值的参数。
9
函数
? 关于形参和实参的说明:
1、形参只有当调用时才临时分配存储单元。
2、实参一定要有确定的值,可以是表达式。
3、实参和形参的类型应相同或赋值相容。
参数的传递是通过调用来完成的,
分为按值传递和按址传递。
10
函数
例 L4_3.c
main( )
{ int a=3,b=5;
void swap( int,int );
swap (a,b);
printf(“a=%d,b=%d\n”,a,b);
}
void swap (int x,int y)
{ int temp;
temp=x; x=y; y=temp;
printf(“x=%d,y=%d \n”,x,y);
}
是按值传递的
按址传递放在指针里面讲。
Main()函数:
调用 Swap函数
传递值
3
5
a
b
3
5
x
y
3
temp
11
函数调用
函数声明
函数定义,其位置
可以在 main()前,
也可以在 main()后
Main()
{ float x,y;
float y(float);
scanf(“%f”,&x);
Printf(“y=%8.2f\n”,y(x));
}
float y (float x )
{ if (x<0)
return (x*x- x+1);
else
return (x*x*x+x+3);
}
12
实参为数组
元素
#include "stdio.h"
main()
{int a[3],i,m;
int max(int,int);
printf("Please input 3 int:\n");
for(i=0;i<3;i++)
scanf("%d",&a[i]);
m=max(a[0],a[1]);
m=max(m,a[2]);
printf("max=%d\n",m);
}
int max(int a,int b)
{return(a>b?a:b);
}
1、数组元素作为函数实际参数
? 数组元素和简单变量具有相同的特性,因此可
以作为函数的实参来出现。( L4_4_0)
形参为整形
13
2、数组名作为函数参数
? 有一个一唯数组 score,内放 10个学
生的分数,求平均成绩。
形参
实参
#define N 10
main()
{float score[N],avg;
float average(float arr[N]);
int i;
printf("Please input %d score:\n",N);
for(i=0;i<N;i++)
scanf("%f",&score[i]);
avg=average(score);
printf("Average score is %7.2f\n",avg);
}
float average(float arr[N])
{int i;
float sum=0.0,aver;
for(i=0;i<N;i++)
sum=sum+arr[i];
aver=sum/N;
return(aver);
}
形参和实
参都用数
组名
15
数组名作为函数参数
? 实参数组和形参数组类型应保持一致。
? 在被调用函数中声明的数组大小是不起
任何作用的。因为 C编译系统对形参数组
大小不作检查,形参数组名只代表一个
地址,在调用时将实参数组的首地址传
到形参数组名,也就是说,形参数组并
不在内存中重新申请数组的空间,而是
和实参数组共占存储单元。
根据需要另设一
个参数传递需要
处理的数组元素
的个数
数组元素要参加处理
的实际个数
#define N 10
main()
{float score[N],avg;
float average(float arr[N]);
int i;
printf("Please input %d score:\n",5);
for(i=0;i<5;i++)
scanf("%f",&score[i]);
avg=average(score,5);
printf("Average score is %7.2f\n",avg);
}
float average(float arr[],int n)
{int i;
float sum=0.0,aver;
for(i=0;i<n;i++)
sum=sum+arr[i];
aver=sum/n;
return(aver);
}
实参为数组

形参定义二维数
组的维数及每一
维的大小,可以
缺省第一维的大

#include "stdio.h"
main()
{int a[3][3]={1,1,1,1,1,1,1,1,1};
int s(int aa[][3]);
clrscr();
printf("sum=%d\n",s(a));
}
int s(int a[][3])
{int sum=0;
int i,j;
for(i=0;i<3;i++)
for(j=0;j<3;j++)
if(i==j||i+j==2) sum+=a[i][j];
return(sum);
}
18
嵌套与递归调用,
嵌套调用:在调用一个函数的过程中又调用另一个函数。
( L4_3_1)
递归调用:一个函数直接或间接地调用本身。
例 l4_4.c 求 n!的值
#include,stdio.h”
long factor ( int );
main( )
{ int i;
scanf(“%d,,&i);
printf(“%d != % ld \n”,i,factor(i));
}
函数
19
函数
long factor (int x)
{ if (x==1 | | x==0) return
(1);
else
return ( x*factor( x- 1));
}
调用 factor(4)
Main()函数 Factor(4)
求 4*factor(3)
Factor(3)
求 3*factor(2)
Factor(2)
求 2*factor(1)
Factor(1)
……..
返回值 1
返回值 2返回值 6返回值 24输出Factor(5)=24
20
a,b的有效范围
C的有效范围
外部变量和局部变量
1、外部变量:
是在函数之外定义的变量,它的有效范围是从定义它的
地方开始,到整个程序结束的任何地方。
2、内部变量:
在一个函数内定义的变量,只在本函数内有效,这种变
量就是局部变量。
Main()
{int a,b;
… …
{int c;
c=a+b;
… …
}
… …
}
21
所以在 main()里不能对变
量 C进行引用。
变量 C是再函数 fun
内部定义的,它就
只再该函数内部有
效。
外部变量和局部变量
fun(int a,int b)
{int c;
c=a>b?a:b;
return c;
}
main()
{int a=7,b=10;
fun(a,b);
printf("max=%d\n",c);
}
int c;
c=fun(a,b);
22
在函数 fun()中,虽然没有
定义变量 a,b,但由于它
们定义在程序的最前面,
是全程变量,凡是在定义
该变量的后面定义的函数
均可以引用它们。
外部变量和局部变量
#include,stdio.h”
int a=3,b=5; /*在函数体外定义的变量 */
Main( )
{ void fun( );
printf(,a= %d,b=%d \n,,a,b);
fun ( );
printf(,a= %d,b=%d \n,,a,b);
}
void fun (void)
{
int c ;
c=a;
a=b;
b=c;
}
例 L4_7.c 交换两个变量的值
23
外部变量和局部变量
2、用 extern声明外部变量,用以扩充外部变量的作用域。
( 1)在一个文件内声明外部变量:如,L4_7_0
main( )

void gx( )

int x,y;
void gy( )

还可用它在不同的程序中使用,如 在 file1.c 中引用 file2.c中定义的
变量 x,y。
extern int x,y;
x,y原作用域
x,y新的作用域
24
变量的存储属性
变量的存储类型有:
register型:寄存器类型
auto 型:自动型(缺省的都是 auto型)
static 型:静态型
一、自动变量
[auto] 数据类型 变量名 [=初值表达式 ], … ;
不作特殊说明的变量都是自动变量 。它用之则建、
用完即撤,仅在函数体内或复合语句内建立和有
效,存储在内存的动态数据区中。 例 L4_5.c
程序一进入 main()函
数,变量 x就被创建,
当程序运行到此局部
范围内(即花括号内)
时创建,用完即撤消,
即释放变量。
当程序运行到函数 prt()
内部时,变量 x才被创建,
离开函数时,就被撤消
main( )
{int x=1;
clrscr();
{ void prt(void);
int x=3;
prt( );
printf("x=%d\n",x);
}
printf("%d\n",x);
}
void prt(void)
{
int x=5;
printf("%d\n",x);
}
总之,auto变量是局部变量,在赋值之前,
其值是不确定的,因而,对同一函数的多次
调用,值不保留,因为存储单元被释放过。
26
Register变量
二,register变量(寄存器变量)
与 auto变量完全相同,只是速度快,存储在 CPU的寄存器中。
#include,stdio.h”
void m_table(void)
{ register int i,j;
for (i=1 ; i<=9 ; i++)
for (j=1 ; j<=i ; j++)
{ printf(“%d * %d =%d,,j,i,j*i);
putchar (( I==j)? ? \n ?, ? \t ? );
}
}
Main()
{ void m_table( );
m_table( );
}
27
Static变量
三,static 变量(静态变量)
auto变量是在程序运行过程中建立的,是动态建立、动态
撤消。而 静态变量是在程序一开始就建立的,不撤消
直到程序结束。因而其值具有可继承性,但它只能在
本 程序 内使用。
结果为,1,2,3
一般需要保留函数上一次的调用结果时,就定义成静态变量。
Void incr(void)
{
static int x=0;
x++;
printf(“%d \n,,x );
}
例 L4_6_1.c
main( )
{ void incr(void);
incr( );
incr( );
incr( );
}
28
上一次的结
果,即 n-1的
阶乘
对静态局部变量赋初值
时在编译时进行的,也
就是整个程序的运行过
程中,它已经有初值,
以后每次调用函数时不
再重新赋初值而保留了
上一次的运行结果。
Static变量
? 例 4_7.c:打印 1到 5的阶乘值
int fac(int n)
{static int f=1;
f=f*n;
return(f);
}
main()
{int i;
for(i=1;i<=5;i++)
printf("%d!=%d\n",i,fac(i));
}
29
#include "stdio.h"
int x,y;
one()
{int a,b;
a=25,b=10;
x=a-b;
y=a+b;
return;
}
main()
{int a,b;
a=9,b=5;
x=a+b;
y=a-b;
one();
printf("%d,%d\n",x,y);
}
#include "stdio.h"
int x,y;
two()
{int a=25,b=10;
int x,y;
a=25,b=10;
x=a-b;
y=a+b;
return;
}
main()
{int a=9,b=5;
x=a+b;
y=a-b;
two();
printf("%d,%d\n",x,y);
}
30
#include "stdio.h"
three()
{extern int x,y;
int a=25,b=10;
x=a-b;
y=a+b;
return;
}
int x,y;
main()
{int a=9,b=5;
x=a+b;
y=a-b;
three();
printf("%d,%d\n",x,y);
}