#include?stdio.h
我們提供的服務(wù)有:成都網(wǎng)站制作、網(wǎng)站建設(shè)、微信公眾號開發(fā)、網(wǎng)站優(yōu)化、網(wǎng)站認證、漢中ssl等。為上1000+企事業(yè)單位解決了網(wǎng)站和推廣的問題。提供周到的售前咨詢和貼心的售后服務(wù),是有科學(xué)管理、有技術(shù)的漢中網(wǎng)站制作公司
//---子函數(shù)聲明---//?
int?func1();
int?func2(int?(*func1)());????????//形參為函數(shù)指針(即指向函數(shù)的指針)?
//---主函數(shù)---//?
int?main(){
printf("向函數(shù)二傳遞函數(shù)一,\n即函數(shù)一作為函數(shù)二的參數(shù)。\n");
printf("%d?",func1());
printf("%d\n",func2(func1));//注意函數(shù)名即為函數(shù)地址?。。。。?!?
}????????????????????????????????//實參為函數(shù)名func1或者func1,兩者等價,而非func1()?
//---子函數(shù)定義---//?
int?func1(){
return?1;
}
int?func2(int?(*func1)()){????????//形參為函數(shù)指針(即指向函數(shù)的指針)
return?func1()+1;
}
Question:在函數(shù)void function()中,需要將另外一個函數(shù)double input()的函數(shù)名作為參數(shù)。
typedef double (*P)(int);
Note:參數(shù)部分需要注意,函數(shù)input()所有的輸入?yún)?shù)類型都需要包含在內(nèi)。
例如 double input(double u[5], int num, double x), 則在定義時寫作 typedef double (*P)(double*,int,double)。
此時,函數(shù)作為一種類型,可以直接被其他函數(shù)調(diào)用。
函數(shù)聲明中定義 void function(P input),調(diào)用function(input)即可。
也可以寫成
※※※※※※※※※※※第一個問題的答復(fù)※※※※※※※※※※※※※※※
看明白了你的意思,
其實不知道你自己有沒有注意到,你所嘗試的方法,
如果用C語言來做的話,其實就是實現(xiàn)了C語言的部分面向?qū)ο蟮膶崿F(xiàn),
說是"部分"的原因是,這僅僅是實現(xiàn)了面向?qū)ο蟮摹胺椒ā薄?/p>
如果想實現(xiàn)的話,準(zhǔn)確的講,應(yīng)該不是你所說的,將"函數(shù)"作為形參,
應(yīng)該是將“函數(shù)指針”作為形參。
這個在回調(diào)(CallBack)函數(shù)設(shè)計時,使用的非常多,
簡單舉一個例子:
#include stdlib.h
#include stdio.h
int Do1()
{
return 0;
}
int Do2(int num)
{
printf("The num is: %d\n", num);
return 0;
}
void CallBack1(void (*ptr)())//指向函數(shù)的指針作函數(shù)參數(shù)
{
(*ptr)();
}
void CallBack2(int n, int (*ptr)())//指向函數(shù)的指針作函數(shù)參數(shù),這里第一個參數(shù)是為指向函數(shù)的指針服務(wù)的,
{ //不能寫成void Caller2(int (*ptr)(int n)),這樣的定義語法錯誤。
(*ptr)(n);
return;
}
int main()
{
CallBack1(Do1); //相當(dāng)于調(diào)用Do1();
CallBack2(50, Do2); //相當(dāng)于調(diào)用Do2(50);
return 0;
}
※※※※※※※※※※※第一個問題的答復(fù)※※※※※※※※※※※※※※※
※※※※※※※※※※※補充問題的答復(fù)※※※※※※※※※※※※※※※
針對你的補充問題,解答如下:
這個是可變形參的實現(xiàn),準(zhǔn)確地說,不是通過數(shù)組實現(xiàn)的,而是通過棧實現(xiàn)的。
C語言中的printf,scanf就是最常見的可變形參函數(shù),定義一個可變形參的函數(shù)很簡單,如void print(int n, ...) ,函數(shù)中對參數(shù)的處理主要是通過對棧進行操作,而c函數(shù)的實參都是自右向左壓入棧的. 主要的棧操作(都是宏)有va_list,va_start ,va_arg,va_end, 定義如下:
typedef char * va_list;
#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) ~(sizeof(int) - 1) )
#define va_start _crt_va_start
#define va_arg _crt_va_arg
#define va_end _crt_va_end
#define _crt_va_start(ap,v) ( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) )
#define _crt_va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
#define _crt_va_end(ap) ( ap = (va_list)0 )
va_start(ap,v):主要是獲取可變參數(shù)列表的首地址,然后賦值給ap,近似ap=v+sizeof(v) (這里暫不考慮內(nèi)存對齊和類型轉(zhuǎn)換)
va_arg(ap,t):取得返回類型t的可變參數(shù)值, 并使ap指向下一個參數(shù): ap += sizeof(t),這里的t是可變參數(shù)的數(shù)據(jù)類型,如int,float之類
va_end(ap):給ap初始化
va_start(ap,v) va_arg(ap,t) va_end(ap)三者合用,保證程序的健壯性.
一個使用可變形參的簡單程序:
#include stdio.h
#include stdarg.h //包含va_list等定義
float sum( float first, ... ) //,...代表可變形參函數(shù)
{
float i=first,sum=0;
va_list maker; //va_list 類型數(shù)據(jù)可以保存函數(shù)的所有參數(shù),做為一個列表一樣保存
va_start(maker,first); //設(shè)置列表的起始位置
while(i!=-1.0)
{
sum+=i;
i=va_arg(maker,float); //返回maker列表的當(dāng)前值,并指向列表的下一個位置
}
return sum;
}
void main(void)
{
printf( "sum is: %f\n", sum( 2.0,8.0,8.5,-1.0 ) ); //函數(shù)調(diào)用
}
※※※※※※※※※※※補充問題的答復(fù)※※※※※※※※※※※※※※※
數(shù)組元素就是下標(biāo)變量,它與普通變量并無區(qū)別。 因此它作為函數(shù)實參使用與普通變量是完全相同的,在發(fā)生函數(shù)調(diào)用時,把作為實參的數(shù)組元素的值傳送給形參,實現(xiàn)單向的值傳送?!纠?-4】說明了這種情況。
【例8-7】判別一個整數(shù)數(shù)組中各元素的值,若大于0 則輸出該值,若小于等于0則輸出0值。編程如下:#include stdio.hvoid nzp(int v){ if(v0) printf("%d ",v); else printf("%d ",0);}int main(void){ int a[5],i; printf("input 5 numbers\n"); for(i=0;i5;i++){ scanf("%d",a[i]); nzp(a[i]); } return 0;}
本程序中首先定義一個無返回值函數(shù)nzp,并說明其形參v為整型變量。在函數(shù)體中根據(jù)v值輸出相應(yīng)的結(jié)果。在main函數(shù)中用一個for語句輸入數(shù)組各元素,每輸入一個就以該元素作實參調(diào)用一次nzp函數(shù),即把a[i]的值傳送給形參v,供nzp函數(shù)使用。
數(shù)組名作為函數(shù)參數(shù)
用數(shù)組名作函數(shù)參數(shù)與用數(shù)組元素作實參有幾點不同。
1) 用數(shù)組元素作實參時,只要數(shù)組類型和函數(shù)的形參變量的類型一致,那么作為下標(biāo)變量的數(shù)組元素的類型也和函數(shù)形參變量的類型是一致的。因此,并不要求函數(shù)的形參也是下標(biāo)變量。換句話說,對數(shù)組元素的處理是按普通變量對待的。用數(shù)組名作函數(shù)參數(shù)時,則要求形參和相對應(yīng)的實參都必須是類型相同的數(shù)組,都必須有明確的數(shù)組說明。當(dāng)形參和實參二者不一致時,即會發(fā)生錯誤。
2) 在普通變量或下標(biāo)變量作函數(shù)參數(shù)時,形參變量和實參變量是由編譯系統(tǒng)分配的兩個不同的內(nèi)存單元。在函數(shù)調(diào)用時發(fā)生的值傳送是把實參變量的值賦予形參變量。在用數(shù)組名作函數(shù)參數(shù)時,不是進行值的傳送,即不是把實參數(shù)組的每一個元素的值都賦予形參數(shù)組的各個元素。因為實際上形參數(shù)組并不存在,編譯系統(tǒng)不為形參數(shù)組分配內(nèi)存。那么,數(shù)據(jù)的傳送是如何實現(xiàn)的呢?在我們曾介紹過,數(shù)組名就是數(shù)組的首地址。因此在數(shù)組名作函數(shù)參數(shù)時所進行的傳送只是地址的傳送,也就是說把實參數(shù)組的首地址賦予形參數(shù)組名。形參數(shù)組名取得該首地址之后,也就等于有了實在的數(shù)組。實際上是形參數(shù)組和實參數(shù)組為同一數(shù)組,共同擁有一段內(nèi)存空間。
上圖說明了這種情形。圖中設(shè)a為實參數(shù)組,類型為整型。a占有以2000為首地址的一塊內(nèi)存區(qū)。b為形參數(shù)組名。當(dāng)發(fā)生函數(shù)調(diào)用時,進行地址傳送,把實參數(shù)組a的首地址傳送給形參數(shù)組名b,于是b也取得該地址2000。于是a,b兩數(shù)組共同占有以2000為首地址的一段連續(xù)內(nèi)存單元。從圖中還可以看出a和b下標(biāo)相同的元素實際上也占相同的兩個內(nèi)存單元(整型數(shù)組每個元素占二字節(jié))。例如a[0]和b[0]都占用2000和2001單元,當(dāng)然a[0]等于b[0]。類推則有a[i]等于b[i]。
【例8-8】數(shù)組a中存放了一個學(xué)生5門課程的成績,求平均成績。#include stdio.hfloat aver(float a[5]){ int i; float av,s=a[0]; for(i=1;i5;i++) s=s+a[i]; av=s/5; return av;}int main(void){ float sco[5],av; int i; printf("\ninput 5 scores:\n"); for(i=0;i5;i++) scanf("%f",sco[i]); av=aver(sco); printf("average score is %5.2f",av); return 0;}
本程序首先定義了一個實型函數(shù)aver,有一個形參為實型數(shù)組a,長度為5。在函數(shù)aver中,把各元素值相加求出平均值,返回給主函數(shù)。主函數(shù)main 中首先完成數(shù)組sco的輸入,然后以sco作為實參調(diào)用aver函數(shù),函數(shù)返回值送av,最后輸出av值。 從運行情況可以看出,程序?qū)崿F(xiàn)了所要求的功能。
3) 前面已經(jīng)討論過,在變量作函數(shù)參數(shù)時,所進行的值傳送是單向的。即只能從實參傳向形參,不能從形參傳回實參。形參的初值和實參相同,而形參的值發(fā)生改變后,實參并不變化,兩者的終值是不同的。而當(dāng)用數(shù)組名作函數(shù)參數(shù)時,情況則不同。由于實際上形參和實參為同一數(shù)組,因此當(dāng)形參數(shù)組發(fā)生變化時,實參數(shù)組也隨之變化。當(dāng)然這種情況不能理解為發(fā)生了“雙向”的值傳遞。但從實際情況來看,調(diào)用函數(shù)之后實參數(shù)組的值將由于形參數(shù)組值的變化而變化。為了說明這種情況,把【例5.4】改為【例5.6】的形式。
【例8-9】題目同【例8.7】。改用數(shù)組名作函數(shù)參數(shù)。#include stdio.hvoid nzp(int a[5]){ int i; printf("\nvalues of array a are:\n"); for(i=0;i5;i++){ if(a[i]0) a[i]=0; printf("%d ",a[i]); }}int main(void){ int b[5],i; printf("\ninput 5 numbers:\n"); for(i=0;i5;i++) scanf("%d",b[i]); printf("initial values of array b are:\n"); for(i=0;i5;i++) printf("%d ",b[i]); nzp(b); printf("\nlast values of array b are:\n"); for(i=0;i5;i++) printf("%d ",b[i]); return 0;}
本程序中函數(shù)nzp的形參為整數(shù)組a,長度為5。主函數(shù)中實參數(shù)組b也為整型,長度也為5。在主函數(shù)中首先輸入數(shù)組b的值,然后輸出數(shù)組b的初始值。然后以數(shù)組名b為實參調(diào)用nzp函數(shù)。在nzp中,按要求把負值單元清0,并輸出形參數(shù)組a的值。 返回主函數(shù)之后,再次輸出數(shù)組b的值。從運行結(jié)果可以看出,數(shù)組b的初值和終值是不同的,數(shù)組b的終值和數(shù)組a是相同的。這說明實參形參為同一數(shù)組,它們的值同時得以改變。
用數(shù)組名作為函數(shù)參數(shù)時還應(yīng)注意以下幾點:
①形參數(shù)組和實參數(shù)組的類型必須一致,否則將引起錯誤。
②形參數(shù)組和實參數(shù)組的長度可以不相同,因為在調(diào)用時,只傳送首地址而不檢查形參數(shù)組的長度。當(dāng)形參數(shù)組的長度與實參數(shù)組不一致時,雖不至于出現(xiàn)語法錯誤(編譯能通過),但程序執(zhí)行結(jié)果將與實際不符,這是應(yīng)予以注意的。
【例8.10】如把例8.9修改如下:#include stdio.hvoid nzp(int a[8]){ int i; printf("\nvalues of array aare:\n"); for(i=0;i8;i++){ if(a[i]0)a[i]=0; printf("%d ",a[i]); }}int main(void){ int b[5],i; printf("\ninput 5 numbers:\n"); for(i=0;i5;i++) scanf("%d",b[i]); printf("initial values of array b are:\n"); for(i=0;i5;i++) printf("%d ",b[i]); nzp(b); printf("\nlast values of array b are:\n"); for(i=0;i5;i++) printf("%d ",b[i]); return 0;}
本程序與【例8.9】程序比,nzp函數(shù)的形參數(shù)組長度改為8,函數(shù)體中,for語句的循環(huán)條件也改為i8。因此,形參數(shù)組a和實參數(shù)組b的長度不一致。編譯能夠通過,但從結(jié)果看,數(shù)組a的元素a[5]、a[6]、a[7]顯然是無意義的。
③在函數(shù)形參表中,允許不給出形參數(shù)組的長度,或用一個變量來表示數(shù)組元素的個數(shù)。例如,可以寫為:
void nzp(int a[])
或?qū)憺?/p>
void nzp( int a[], int n )
其中形參數(shù)組a沒有給出長度,而由n值動態(tài)地表示數(shù)組的長度。n的值由主調(diào)函數(shù)的實參進行傳送。由此,【例8-10】又可改為【例8-11】的形式。
【例8-11】復(fù)制純文本新窗口
#include stdio.hvoid nzp(int a[],int n){ int i; printf("\nvalues of array a are:\n"); for(i=0;in;i++){ if(a[i]0) a[i]=0; printf("%d ",a[i]); }}int main(void){ int b[5],i; printf("\ninput 5 numbers:\n"); for(i=0;i5;i++) scanf("%d",b[i]); printf("initial values of array b are:\n"); for(i=0;i5;i++) printf("%d ",b[i]); nzp(b,5); printf("\nlast values of array b are:\n"); for(i=0;i5;i++) printf("%d ",b[i]); return 0;}
網(wǎng)站欄目:c語言可以用函數(shù)作為參數(shù) c語言可以用函數(shù)作為參數(shù)嘛
分享地址:http://jinyejixie.com/article42/dohhiec.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站營銷、品牌網(wǎng)站制作、、ChatGPT、全網(wǎng)營銷推廣、服務(wù)器托管
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)