學(xué)習(xí)啦 > 知識(shí)大全 > 知識(shí)百科 > 百科知識(shí) > C語(yǔ)言程序中什么是函數(shù)

C語(yǔ)言程序中什么是函數(shù)

時(shí)間: 思行858 分享

C語(yǔ)言程序中什么是函數(shù)

  C語(yǔ)言的函數(shù)問(wèn)題是困擾很多學(xué)者的問(wèn)題的,c程序中也有一定的函數(shù)的。那么下面一起來(lái)看看學(xué)習(xí)啦小編為大家精心推薦的c程序中的函數(shù),希望能夠?qū)δ兴鶐椭?/p>

  C語(yǔ)言讀書(shū)筆記:函數(shù)

  先來(lái)看看函數(shù)的一般形式,嘗試寫(xiě)一個(gè)加法的函數(shù):

  思路是這樣的:首先得有頭文件,頭文件之后就得寫(xiě)主函數(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ù)的定義直接寫(xiě)在主函數(shù)之前。

  函數(shù)首部int add(int a, int b)中的第一個(gè)int,即add之前的這個(gè)int稱(chēng)為函數(shù)的類(lèi)型。表明這個(gè)函數(shù)將要返回一個(gè)整數(shù)類(lèi)型的值。這個(gè)類(lèi)型可以是C語(yǔ)言中任何被允許的數(shù)據(jù)類(lèi)型,包括void,意為無(wú)返回值類(lèi)型,即這個(gè)函數(shù)不需要返回任何的值。

  函數(shù)首部int add(int a, int b)中的add稱(chēng)為函數(shù)的名字,簡(jiǎn)稱(chēng)函數(shù)名。

  函數(shù)首部int add(int a, int b)中int a和int b稱(chēng)為函數(shù)的形式參數(shù)。這里形式參數(shù)理論上可以有無(wú)窮多個(gè),當(dāng)然,現(xiàn)實(shí)情況下3-5個(gè)就已經(jīng)算是很多了;形式參數(shù)中,即使a和b都是int類(lèi)型的,也要分別定義才行;形式參數(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ù)的類(lèi)型和返回值的類(lèi)型必須嚴(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稱(chēng)為實(shí)際參數(shù),即你究竟想讓函數(shù)幫你計(jì)算哪兩個(gè)數(shù)的加法結(jié)果,你就在這個(gè)括號(hào)里邊寫(xiě)哪幾個(gè)數(shù)字。必須要嚴(yán)格遵守的規(guī)定:實(shí)際參數(shù)和形式參數(shù)必須一一對(duì)應(yīng),數(shù)量應(yīng)該相同,類(lèi)型也保持一致。

  理解了這幾點(diǎn)之后,一個(gè)基本的函數(shù)就已經(jīng)可以寫(xiě)出來(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類(lèi)型的e對(duì)應(yīng)%lf,記住不要缺少&

  printf("pi=%lf\n", f_pi(e) ); // 函數(shù)返回的是個(gè)double類(lèi)型的值,直接輸出

  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) //滿(mǎn)足條件就循環(huán)

  {

  item=flag*1.0/denominator; //計(jì)算每一項(xiàng)的值。flag控制符號(hào)

  //1.0必須寫(xiě)出小數(shù)位,否則整項(xiàng)就變成一個(gè)整型值

  sum+=item; //累加

  flag = -flag; //符號(hào)正負(fù)切換

  denominator = denominator + 2;//分母遞增

  }

  return sum; //sum的類(lèi)型和函數(shù)的類(lèi)型必須一致

  }

  函數(shù)的定義和調(diào)用其實(shí)并不難理解,相信很多人困擾在參數(shù)的傳遞上,接下來(lái)總結(jié)一下函數(shù)參數(shù)傳遞的幾種方式:

  正常的參數(shù)調(diào)用,例如int、float、double等一一對(duì)應(yīng)的傳遞。

  無(wú)參數(shù),也無(wú)返回值。例如下列代碼就只是為了輸出一些語(yǔ)句。這種做法在語(yǔ)法上是被允許的,但是并不推薦這么寫(xiě)。

  void printf()

  {

  printf("hello world!");

  }

  3. 參數(shù)是數(shù)組的名字。我們知道數(shù)組的名字是個(gè)地址,那么如果實(shí)參是數(shù)組名的話(huà),我們可以將形參設(shè)置成指針,指向?qū)崊鬟f過(guò)來(lái)的數(shù)組的首地址。

  4. 參數(shù)是指針。如果實(shí)參是指針,那么形參肯定也得是指針。保持類(lèi)型一致即可,然后在函數(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ù),直接都寫(xiě)在main函數(shù)中多方便?

  非也!

  C語(yǔ)言是一個(gè)過(guò)程化的語(yǔ)言,C語(yǔ)言中的主函數(shù)其實(shí)是用來(lái)主導(dǎo)程序的進(jìn)程和數(shù)據(jù)的流動(dòng)方向的。如果將主函數(shù)寫(xiě)的過(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è)有這樣一種情況,我們要編寫(xiě)一個(gè)庫(kù),它提供了某些排序算法的實(shí)現(xiàn),如冒泡排序、快速排序、shell排序、shake排序等等,但為使庫(kù)更加通用,不想在函數(shù)中嵌入排序邏輯,而讓使用者來(lái)實(shí)現(xiàn)相應(yīng)的邏輯;或者,想讓庫(kù)可用于多種數(shù)據(jù)類(lèi)型(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ā)生。

  下面是自己寫(xiě)的一個(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)行物理寫(xiě)入時(shí),應(yīng)該使用這個(gè)函數(shù)。例如調(diào)用fflush函數(shù)保證調(diào)試信息實(shí)際打印出來(lái),而不是保存在緩沖區(qū)中直到以后才打印。

  二.定位函數(shù)

  1.在正常情況下,數(shù)據(jù)以線(xiàn)性的方式寫(xiě)入,這意味著后面寫(xiě)入的數(shù)據(jù)在文件中的位置是在以前所有寫(xiě)入數(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ū)懭氩僮鞯奈恢?。它的?1個(gè)參數(shù)是需要改變的流。它的第2和第3個(gè)參數(shù)標(biāo)識(shí)文件中需要定位的位置。

  1>試圖定位到一個(gè)文件的起始位置之前是一個(gè)錯(cuò)誤。定位到文件尾并進(jìn)行寫(xiě)入將擴(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>最后,定位允許你從寫(xiě)入模式切換到讀取模式,或者回到打開(kāi)的流以便更新。


看了“c程序中什么是函數(shù)”的人還看了:

1.c語(yǔ)言get函數(shù)的用法

2.c程序員筆試題

3.常用c/c++的面試題有些呢

4.c語(yǔ)言中free的用法

5.c語(yǔ)言函數(shù)if的用法

6.c語(yǔ)言中default的用法

7.關(guān)于c語(yǔ)言的面試題

1327881