c语言void nzp,C语言函数是什么

C语言函数是什么

二、数组名作为函数参数

用数组名作函数参数与用数组元素作实参有几点不同:

1. 用数组元素作实参时,只要数组类型和函数的形参变量的类型一致,那么作为下标变量的数组元素的类型也和函数形参变量的类型是一致的。因此, 并不要求函数的形参也是下标变量。 换句话说,对数组元素的处理是按普通变量对待的。用数组名作函数参数时, 则要求形参和相对应的实参都必须是类型相同的数组,都必须有明确的数组说明。当形参和实参二者不一致时,即会发生错误。 2. 在普通变量或下标变量作函数参数时,形参变量和实参变量是由编译系统分配的两个不同的内存单元。在函数调用时发生的值传送是把实参变量的值赋予形参变量。在用数组名作函数参数时,不是进行值的传送,即不是把实参数组的每一个元素的值都赋予形参数组的各个元素。因为实际上形参数组并不存在,编译系统不为形参数组分配内存。那么,数据的传送是如何实现的呢? 在第四章中我们曾介绍过,数组名就是数组的首地址。因此在数组名作函数参数时所进行的传送只是地址的传送, 也就是说把实参数组的首地址赋予形参数组名。形参数组名取得该首地址之后,也就等于有了实在的数组。实际上是形参数组和实参数组为同一数组,共同拥有一段内存空间。图5.1说明了这种情形。图中设a为实参数组,类型为整型。a占有以2000 为首地址的一块内存区。b为形参数组名。当发生函数调用时,进行地址传送, 把实参数 组a的首地址传送给形参数组名b,于是b也取得该地址2000。 于是a,b两数组共同占有以2000 为首地址的一段连续内存单元。从图中还可以看出a和b下标相同的元素实际上也占相同的两个内

存单元(整型数组每个元素占二字节)。例如a[0]和b[0]都占用2000和2001单元,当然a[0]等于b[0]。类推则有a[i]等于b[i]。

[例5.5]数组a中存放了一个学生5门课程的成绩,求平均成绩。float aver(float a[5])

{

int i;

float av,s=a[0];

for(i=1;i<5;i++)

s=s+a[i];

av=s/5;

return av;

}

void main()

{

float sco[5],av;

int i;

printf("

input 5 scores:

");

for(i=0;i<5;i++)

scanf("%f",&sco[i]);

av=aver(sco);

printf("average score is %5.2f",av);

}

float aver(float a[5])

{ ……

}

void main()

{

……

for(i=0;i<5;i++)

scanf("%f",&sco[i]);

av=aver(sco);

……

}

本程序首先定义了一个实型函数aver,有一个形参为实型数组a,长度为5。在函数aver中,把各元素值相加求出平均值,返回给主函数。主函数main 中首先完成数组sco的输入,然后以sco作为实参调用aver函数,函数返回值送av,最后输出av值。 从运行情况可以看出,程序实现了所要求的功能

3. 前面已经讨论过,在变量作函数参数时,所进行的值传送是单向的。即只能从实参传向形参,不能从形参传回实参。形参的初值和实参相同, 而形参的值发生改变后,实参并不变化, 两者的终值是不同的。例5.3证实了这个结论。 而当用数组名作函数参数时,情况则不同。 由于实际上形参和实参为同一数组, 因此当形参数组发生变化时,实参数组也随之变化。 当然这种情况不能理解为发生了“双向”的值传递。但从实际情况来看,调用函数之后实参数组的值将由于形参数组值的变化而变化。为了说明这种情况,把例5.4改为例5.6的形式。[例5.6]题目同5.4例。改用数组名作函数参数。

void nzp(int a[5])

{

int i;

printf("

values of array a are:

");

for(i=0;i<5;i++)

{

if(a[i]<0) a[i]=0;

printf("%d ",a[i]);

}

}

main()

{

int b[5],i;

printf("

input 5 numbers:

");

for(i=0;i<5;i++)

scanf("%d",&b[i]);

printf("initial values of array b are:

");

for(i=0;i<5;i++)

printf("%d ",b[i]);

nzp(b);

printf("

last values of array b are:

");

for(i=0;i<5;i++)

printf("%d ",b[i]);

}

void nzp(int a[5])

{ ……

}

main()

{

int b[5],i;

……

nzp(b);

……

}

本程序中函数nzp的形参为整数组a,长度为 5。 主函数中实参数组b也为整型,长度也为5。在主函数中首先输入数组b的值,然后输出数组b的初始值。 然后以数组名b为实参调用nzp函数。在nzp中,按要求把负值单元清0,并输出形参数组a的值。 返回主函数之后,再次输出数组b的值。从运行结果可以看出,数组b 的初值和终值是不同的,数组b 的终值和数组a是相同的。这说明实参形参为同一数组,它们的值同时得以改变。 用数组名作为函数参数时还应注意以下几点:

a. 形参数组和实参数组的类型必须一致,否则将引起错误。

b. 形参数组和实参数组的长度可以不相同,因为在调用时,只传送首地址而不检查形参数组的长度。当形参数组的长度与实参数组不一致时,虽不至于出现语法错误(编译能通过),但程序执行结果将与实际不符,这是应予以注意的。如把例5.6修改如下:

void nzp(int a[8])

{

int i;

printf("

values of array aare:

");

for(i=0;i<8;i++)

{

if(a[i]<0)a[i]=0;

printf("%d",a[i]);

}

}

main()

{

int b[5],i;

printf("

input 5 numbers:

");

for(i=0;i<5;i++)

scanf("%d",&b[i]);

printf("initial values of array b are:

");

for(i=0;i<5;i++)

printf("%d",b[i]);

nzp(b);

printf("

last values of array b are:

");

for(i=0;i<5;i++)

printf("%d",b[i]);

}

本程序与例5.6程序比,nzp函数的形参数组长度改为8,函数体中,for语句的循环条件也改为i<8。因此,形参数组 a和实参数组b的长度不一致。编译能够通过,但从结果看,数组a的元素a[5],a[6],a[7]显然是无意义的。c. 在函数形参表中,允许不给出形参数组的长度,或用一个变量来表示数组元素的个数。

例如:可以写为:

void nzp(int a[])

或写为

void nzp(int a[],int n)

其中形参数组a没有给出长度,而由n值动态地表示数组的长度。n的值由主调函数的实参进行传送。

由此,例5.6又可改为例5.7的形式。

[例5.7]void nzp(int a[],int n)

{

int i;

printf("

values of array a are:

");

for(i=0;i

{

if(a[i]<0) a[i]=0;

printf("%d ",a[i]);

}

}

main()

{

int b[5],i;

printf("

input 5 numbers:

");

for(i=0;i<5;i++)

scanf("%d",&b[i]);

printf("initial values of array b are:

");

for(i=0;i<5;i++)

printf("%d ",b[i]);

nzp(b,5);

printf("

last values of array b are:

");

for(i=0;i<5;i++)

printf("%d ",b[i]);

}

void nzp(int a[],int n)

{ ……

}

main()

{

……

nzp(b,5);

……

}

本程序nzp函数形参数组a没有给出长度,由n 动态确定该长度。在main函数中,函数调用语句为nzp(b,5),其中实参5将赋予形参n作为形参数组的长度。

d. 多维数组也可以作为函数的参数。 在函数定义时对形参数组可以指定每一维的长度,也可省去第一维的长度。因此,以下写法都是合法的。

int MA(int a[3][10])

int MA(int a[][10])

函数的嵌套调用

C语言中不允许作嵌套的函数定义。因此各函数之间是平行的,不存在上一级函数和下一级函数的问题。 但是C语言允许在一个函数的定义中出现对另一个函数的调用。 这样就出现了函数的嵌套调用。即在被调函数中又调用其它函数。 这与其它语言的子程序嵌套的情形是类似的。其关系可表示如图5.2。

图5.2表示了两层嵌套的情形。其执行过程是:执行main函数中调用a函数的语句时,即转去执行a函数,在a函数中调用b 函数时,又转去执行b函数,b函数执行完毕返回a函数的断点继续执行,a 函数执行完毕返回main函数的断点继续执行。

[例5.8]计算s=2?2!+3?2!

本题可编写两个函数,一个是用来计算平方值的函数f1, 另一个是用来计算阶乘值的函数f2。主函数先调f1计算出平方值, 再在f1中以平方值为实参,调用 f2计算其阶乘值,然后返回f1,再返回主函数,在循环程序中计算累加和。

long f1(int p)

{

int k;

long r;

long f2(int);

k=p*p;

r=f2(k);

return r;

}

long f2(int q)

{

long c=1;

int i;

for(i=1;i<=q;i++)

c=c*i;

return c;

}

main()

{

int i;

long s=0;

for (i=2;i<=3;i++)

s=s+f1(i);

printf("

s=%ld

",s);

}

long f1(int p)

{

……

long f2(int);

r=f2(k);

……

}

long f2(int q)

{

……

}

main()

{ ……

s=s+f1(i);

……

}

在程序中,函数f1和f2均为长整型,都在主函数之前定义, 故不必再在主函数中对f1和f2加以说明。在主程序中, 执行循环程序依次把i值作为实参调用函数f1求i?2值。在f1中又发生对函数f2的调用,这时是把i?2的值作为实参去调f2,在f2 中完成求i?2! 的计算。f2执行完毕把C值(即i?2!)返回给f1,再由f1 返回主函数实现累加。至此,由函数的嵌套调用实现了题目的要求。 由于数值很大, 所以函数和一些变量的类型都说明为长整型,否则会造成计算错误。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值