C語(yǔ)言程序中什么是函數(shù)
C語(yǔ)言程序中什么是函數(shù)
C語(yǔ)言的函數(shù)問(wèn)題是困擾很多學(xué)者的問(wèn)題的,c程序中也有一定的函數(shù)的。那么下面一起來(lái)看看學(xué)習(xí)啦小編為大家精心推薦的c程序中的函數(shù),希望能夠?qū)δ兴鶐椭?/p>
C語(yǔ)言讀書筆記:函數(shù)
先來(lái)看看函數(shù)的一般形式,嘗試寫一個(gè)加法的函數(shù):
思路是這樣的:首先得有頭文件,頭文件之后就得寫主函數(shù),主函數(shù)的內(nèi)部應(yīng)該就是加法的過(guò)程,我們將所有加法的語(yǔ)句都拿出來(lái)組成一個(gè)函數(shù)。代碼如下:
#include <stdio.h>
int add(int a, int b);
int main()
{
int result = add(3,5);
printf("sum is %d\n", result);
return 0;
}
int add(int a, int b)
{
int sum;
sum = a+b;
return sum;
}
這是一個(gè)最簡(jiǎn)單的函數(shù),描述了一個(gè)加法函數(shù)的定義和調(diào)用的過(guò)程。
int add(int a, int b) 成為函數(shù)的首部。
有了首部之后,就得考慮一件事情,將首部復(fù)制之后,加上一個(gè)分號(hào),粘貼在主函數(shù)之前,作為函數(shù)的原型聲明。試想,我們?cè)谥骱瘮?shù)里邊是不是要先定義變量result才能使用result?那么函數(shù)的道理也是一樣的,當(dāng)程序運(yùn)行到主函數(shù)中語(yǔ)句“int result = add(3,5);”的時(shí)候,如果向上沒(méi)有尋找到add()的定義,那么編譯器一定就會(huì)報(bào)錯(cuò)。所以要不然添加函數(shù)的原型聲明,要不然就將函數(shù)的定義直接寫在主函數(shù)之前。
函數(shù)首部int add(int a, int b)中的第一個(gè)int,即add之前的這個(gè)int稱為函數(shù)的類型。表明這個(gè)函數(shù)將要返回一個(gè)整數(shù)類型的值。這個(gè)類型可以是C語(yǔ)言中任何被允許的數(shù)據(jù)類型,包括void,意為無(wú)返回值類型,即這個(gè)函數(shù)不需要返回任何的值。
函數(shù)首部int add(int a, int b)中的add稱為函數(shù)的名字,簡(jiǎn)稱函數(shù)名。
函數(shù)首部int add(int a, int b)中int a和int b稱為函數(shù)的形式參數(shù)。這里形式參數(shù)理論上可以有無(wú)窮多個(gè),當(dāng)然,現(xiàn)實(shí)情況下3-5個(gè)就已經(jīng)算是很多了;形式參數(shù)中,即使a和b都是int類型的,也要分別定義才行;形式參數(shù)可以在函數(shù)中直接使用,無(wú)須再次定義;形式參數(shù)是用來(lái)告訴調(diào)用者,你應(yīng)該給我傳遞來(lái)什么樣子的數(shù)據(jù),我好利用你給我的數(shù)據(jù)在函數(shù)中進(jìn)行計(jì)算。
int add(int a, int b){}中的{}就是函數(shù)體的內(nèi)容了。函數(shù)需要進(jìn)行的所有的操作都要放在這對(duì)大括號(hào)中。想必大家也看到了函數(shù)體中最后有一條語(yǔ)句是return,這條語(yǔ)句起到的作用就是返回函數(shù)計(jì)算的結(jié)果,在這個(gè)程序中就是將加法的結(jié)果返回給主函數(shù)。需要注意的是,函數(shù)的類型和返回值的類型必須嚴(yán)格一致!
函數(shù)的定義到此為止,接下來(lái)講講函數(shù)的調(diào)用方式。只要定義好函數(shù),通過(guò)函數(shù)名(實(shí)際參數(shù)1,實(shí)際參數(shù)2,實(shí)際參數(shù)n)這種方式就可以調(diào)用函數(shù)了。例如主函數(shù)中的“int result = add(3,5);”,就是調(diào)用了add函數(shù)。這里,3和5稱為實(shí)際參數(shù),即你究竟想讓函數(shù)幫你計(jì)算哪兩個(gè)數(shù)的加法結(jié)果,你就在這個(gè)括號(hào)里邊寫哪幾個(gè)數(shù)字。必須要嚴(yán)格遵守的規(guī)定:實(shí)際參數(shù)和形式參數(shù)必須一一對(duì)應(yīng),數(shù)量應(yīng)該相同,類型也保持一致。
理解了這幾點(diǎn)之后,一個(gè)基本的函數(shù)就已經(jīng)可以寫出來(lái)了。接下來(lái)來(lái)個(gè)題目嘗試一下:
輸入精度e,使用公式求π的近似值,精確到最后一項(xiàng)的絕對(duì)值小于e。公式:π=1-1/3+1/5-1/7+...
代碼:
//首先得有頭文件
#include <stdio.h>
#include <math.h> //后邊要使用到fabs絕對(duì)值函數(shù)
//然后就是主函數(shù)了
int main(void)
{
double pi, e; //定義所需變量
double f_pi(double e); //原型聲明。函數(shù)名只要符合命名規(guī)則即可 //因?yàn)橐笮∮趀,所以也將這個(gè)e傳遞過(guò)去
printf("enter e: "); //輸入的提示
scanf("%lf", &e); // double類型的e對(duì)應(yīng)%lf,記住不要缺少&
printf("pi=%lf\n", f_pi(e) ); // 函數(shù)返回的是個(gè)double類型的值,直接輸出
return 0;
}
double f_pi(double e) //函數(shù)首部,形參和實(shí)參一定要對(duì)應(yīng),可以重名
{
int denominator, flag;
double item, sum;
//請(qǐng)注意“先定義,然后賦初值再使用”的好習(xí)慣!!!
flag = 1; //負(fù)責(zé)變換正負(fù)符號(hào)的變量
denominator = 1; //分母初值為1,第一項(xiàng)的1為1/1
item=1.0; //存放每一項(xiàng)的值
sum=0;
while(fabs(item)>=e) //滿足條件就循環(huán)
{
item=flag*1.0/denominator; //計(jì)算每一項(xiàng)的值。flag控制符號(hào)
//1.0必須寫出小數(shù)位,否則整項(xiàng)就變成一個(gè)整型值
sum+=item; //累加
flag = -flag; //符號(hào)正負(fù)切換
denominator = denominator + 2;//分母遞增
}
return sum; //sum的類型和函數(shù)的類型必須一致
}
函數(shù)的定義和調(diào)用其實(shí)并不難理解,相信很多人困擾在參數(shù)的傳遞上,接下來(lái)總結(jié)一下函數(shù)參數(shù)傳遞的幾種方式:
正常的參數(shù)調(diào)用,例如int、float、double等一一對(duì)應(yīng)的傳遞。
無(wú)參數(shù),也無(wú)返回值。例如下列代碼就只是為了輸出一些語(yǔ)句。這種做法在語(yǔ)法上是被允許的,但是并不推薦這么寫。
void printf()
{
printf("hello world!");
}
3. 參數(shù)是數(shù)組的名字。我們知道數(shù)組的名字是個(gè)地址,那么如果實(shí)參是數(shù)組名的話,我們可以將形參設(shè)置成指針,指向?qū)崊鬟f過(guò)來(lái)的數(shù)組的首地址。
4. 參數(shù)是指針。如果實(shí)參是指針,那么形參肯定也得是指針。保持類型一致即可,然后在函數(shù)內(nèi)部再對(duì)指針進(jìn)行操作。
5. 參數(shù)是結(jié)構(gòu)體。如果實(shí)參是結(jié)構(gòu)體,一般來(lái)說(shuō)我們使用結(jié)構(gòu)體指針來(lái)做形參比較合適。
還是在此分割一下吧,說(shuō)了這么多,可能很多人在問(wèn)問(wèn)什么函數(shù)定義這么麻煩,還要定義函數(shù),直接都寫在main函數(shù)中多方便?
非也!
C語(yǔ)言是一個(gè)過(guò)程化的語(yǔ)言,C語(yǔ)言中的主函數(shù)其實(shí)是用來(lái)主導(dǎo)程序的進(jìn)程和數(shù)據(jù)的流動(dòng)方向的。如果將主函數(shù)寫的過(guò)于復(fù)雜,我們閱讀程序的結(jié)構(gòu)就會(huì)非常的費(fèi)力。
C語(yǔ)言中的函數(shù)回調(diào)
什么是回調(diào)函數(shù)?
簡(jiǎn)而言之,回調(diào)函數(shù)就是一個(gè)通過(guò)函數(shù)指針調(diào)用的函數(shù)。如果你把函數(shù)的指針(地址)作為參數(shù)傳遞給另一個(gè)函數(shù),當(dāng)這個(gè)指針被用為調(diào)用它所指向的函數(shù)時(shí),我們就說(shuō)這是回調(diào)函數(shù)。
為什么要使用回調(diào)函數(shù)?
因?yàn)榭梢园颜{(diào)用者與被調(diào)用者分開(kāi)。調(diào)用者不關(guān)心誰(shuí)是被調(diào)用者,所有它需知道的,只是存在一個(gè)具有某種特定原型、某些限制條件(如返回值為int)的被調(diào)用函數(shù)。
如果想知道回調(diào)函數(shù)在實(shí)際中有什么作用,先假設(shè)有這樣一種情況,我們要編寫一個(gè)庫(kù),它提供了某些排序算法的實(shí)現(xiàn),如冒泡排序、快速排序、shell排序、shake排序等等,但為使庫(kù)更加通用,不想在函數(shù)中嵌入排序邏輯,而讓使用者來(lái)實(shí)現(xiàn)相應(yīng)的邏輯;或者,想讓庫(kù)可用于多種數(shù)據(jù)類型(int、float、string),此時(shí),該怎么辦呢?可以使用函數(shù)指針,并進(jìn)行回調(diào)。
回調(diào)可用于通知機(jī)制,例如,有時(shí)要在程序中設(shè)置一個(gè)計(jì)時(shí)器,每到一定時(shí)間,程序會(huì)得到相應(yīng)的通知,但通知機(jī)制的實(shí)現(xiàn)者對(duì)我們的程序一無(wú)所知。而此時(shí),就需有一個(gè)特定原型的函數(shù)指針,用這個(gè)指針來(lái)進(jìn)行回調(diào),來(lái)通知我們的程序事件已經(jīng)發(fā)生。
下面是自己寫的一個(gè)簡(jiǎn)單的回調(diào)函數(shù),相比其他的那些復(fù)雜的代碼,這個(gè)更容易理解:
#include<stdio.h>
#include<stdlib.h>
void perfect(int n)
{
int i=1;
int count=0;
for(i=1;i<n;i++)
{
if(0==n%i)
{
count+=i;
}
}
if(count==n)
printf("%d是完數(shù)\n",n);
else printf("%d不是完數(shù)\n",n);
}
void myCallback(void (*perfect)(int ),int n)
{
perfect(n);
}
int main()
{
int n;
printf("請(qǐng)輸入一個(gè)正整數(shù)\n");
scanf("%d",&n);
myCallback(perfect,n);
return 0;
}
C語(yǔ)言中的刷新和定位函數(shù)
一.fflush
1.fflush的原型如下:
intfflush(FILE *stream);
2.當(dāng)需要立即把輸出緩沖區(qū)的數(shù)據(jù)進(jìn)行物理寫入時(shí),應(yīng)該使用這個(gè)函數(shù)。例如調(diào)用fflush函數(shù)保證調(diào)試信息實(shí)際打印出來(lái),而不是保存在緩沖區(qū)中直到以后才打印。
二.定位函數(shù)
1.在正常情況下,數(shù)據(jù)以線性的方式寫入,這意味著后面寫入的數(shù)據(jù)在文件中的位置是在以前所有寫入數(shù)據(jù)的后面。C同時(shí)支持隨機(jī)訪問(wèn)I/O,也就是以任意順序訪問(wèn)文件的不同位置。隨機(jī)訪問(wèn)是通過(guò)在讀取或?qū)懭肭?,先定位到文件中需要的位置?lái)實(shí)現(xiàn)的。
2.定位函數(shù)原型:
1>long ftell(FILE*stream);
2>intfseek(FILE *steam,long offset,intfrom);
3.ftell函數(shù)返回流的當(dāng)前位置。即:下一個(gè)讀取或?qū)懭雽⒁_(kāi)始的位置距離文件起始位置的偏移量。該函數(shù)允許保存一個(gè)文件的當(dāng)前位置。
1>在二進(jìn)制流中,這個(gè)值就是當(dāng)前位置距離文件起始位置之間的字節(jié)數(shù)。
2>在文本流中,這個(gè)值表示一個(gè)位置,但它并不一定準(zhǔn)確地表示當(dāng)前位置和文件起始位置之間的字符數(shù),因?yàn)橛行┫到y(tǒng)將對(duì)行末字符進(jìn)行翻譯轉(zhuǎn)換。但是,ftell函數(shù)返回的值總是可以用于fseek函數(shù)中,作為一個(gè)距離文件起始位置的偏移量。
4.fseek函數(shù)允許你一個(gè)流中定位。這個(gè)函數(shù)將改變下一個(gè)讀取或?qū)懭氩僮鞯奈恢谩K牡?1個(gè)參數(shù)是需要改變的流。它的第2和第3個(gè)參數(shù)標(biāo)識(shí)文件中需要定位的位置。
1>試圖定位到一個(gè)文件的起始位置之前是一個(gè)錯(cuò)誤。定位到文件尾并進(jìn)行寫入將擴(kuò)展這個(gè)文件。定位到文件尾之后并進(jìn)行讀取將導(dǎo)致返回一條“到達(dá)文件尾”的信息。
2>在二進(jìn)制流中,從SEEK_END進(jìn)行定位可能不被支持,所以應(yīng)該避免。
3>在文本流中,如果from是SEEK_CUR或SEEK_END,offset必須是零。如果from是SEEK_SET,offset必須是一個(gè)從同一個(gè)流中以前調(diào)用ftell所返回的值。
5.用fseek改變一個(gè)流的位置會(huì)帶來(lái)三個(gè)副作用。
1>首先,行末指示字符被清除。
2>其次,如果在fseek之前使用ungetc把一個(gè)字符返回到流中,那么這個(gè)被退回的字符會(huì)被丟棄,因?yàn)樵诙ㄎ徊僮饕院?,它不再?ldquo;下一個(gè)字符”。
3>最后,定位允許你從寫入模式切換到讀取模式,或者回到打開(kāi)的流以便更新。
看了“c程序中什么是函數(shù)”的人還看了:
2.c程序員筆試題