糯米文學吧

位置:首頁 > 計算機 > C語言

C語言函數的基本學習教程

C語言1.57W

C 語言中的函數等價於 Fortran 語言中的子程序或函數,也等價於 Pascal 語言中的過程或函數。函數為計算的封裝提供了一種簡便的方法,此後使用函數時不需要考慮它是如何實現的。使用設計正確的函數,程序員無需考慮功能是如何實現的,而只需知道它具有哪些功能就夠了。在 C 語言中可以簡單、方便、高效地使用函數。我們經常會看到在定義後僅調用了一次的短函數,這樣做可以使代碼段更清晰易讀。

C語言函數的基本學習教程

到目前為止,我們所使用的函數(如 printf、getchar 和 putchar 等)都是函數庫中提供的函數。現在,讓我們自己動手來編寫一些函數。C 語言沒有像 Fortran 語言一樣提供類似於**的求冪運算符,我們現在通過編寫一個求冪的函數 power(m, n)來説明函數定義的方法。power(m, n)函數用於計算整數 m 的 n 次冪,其中 n 是正整數。對函數調用 power(2,5)來説,其結果值為 32。該函數並非一個實用的求冪函數,它只能處理較小的整數的正整數次冪,但這對於説明問題已足夠了。(標準庫中提供了一個計算 xy 的函數 pow(x, y)。)

下面是函數 power(m, n)的定義及調用它的主程序,這樣我們可以看到一個完整的程序結構。

#includeint power(int m, int n);/* test power function */main(){ int i; for (i = 0; i < 10; ++i) printf("%d %d %dn", i, power(2,i), power(-3,i)); return 0;}/* power: raise base to n-th power; n >= 0 */int power(int base, int n){ int i, p; p = 1; for (i = 1; i <= n; ++i) p = p * base; return p;}

函數定義的一般形式為:

返回值類型 函數名(0 個或多個參數聲明){ 聲明部分 語句序列}

函數定義可以以任意次序出現在一個源文件或多個源文件中,但同一函數不能分割存放在多個文件中。如果源程序分散在多個文件中,那麼,在編譯和加載時,就需要做更多的工作,但這是操作系統的`原因,並不是語言的屬性決定的。我們暫且假定將 main 和 power 這兩個函數放在同一文件中,這樣前面所學的有關運行 C 語言程序的知識仍然有效。

main 函數在下列語句中調用了兩次 power 函數:printf("%d %d %dn", i, power(2, i), power(-i, 3)); 每次調用時,main 函數向 power 函數傳遞兩個參數;在調用執行完成時,power 函數向 main 函數返回一個格式化的整數並打印。在表達式中,power(2, i)同 2 和 i 一樣都是整數

power 函數的第一行語句 int power(int base, int n) 聲明參數的類型、名字以及該函數返回結果的類型。power 函數的參數使用的名字只在 power 函數內部有效,對其它任何函數都是不可見的:其它函數可以使用與之相同的參數名字而不會引起衝突。變量 i 與 p 也是這樣:power 函數中的 i 與 main 函數中的 i 無關。

我們通常把函數定義中圓括號內列表中出現的變量稱為形式參數,而把函數調用中與形式參數對應的值稱為實際參數。

power 函數計算所得的結果通過 return 語句返回給 main 函數。關鍵字 return 的後面可以跟任何表達式,形式為: return 表達式;

函數不一定都有返回值。不帶表達式的 return 語句將把控制權返回給調用者,但不返回有用的值。這等同於在到達函數的右終結花括號時,函數就“到達了盡頭”。主調函數也可以忽略函數返回的值。

讀者可能已經注意到,main 函數的末尾有一個 return 語句。由於 main 本身也是函數,因此也可以向其調用者返回一個值,該調用者實際上就是程序的執行環境。一般來説,返回值為 0 表示正常終止,返回值為非 0 表示出現異常情況或出錯結束條件。為簡潔起見,前面的 main 函數都省略了 return 語句,但我們將在以後的 main 函數中包含 return 語句,以提醒大家注意,程序還要向其執行環境返回狀態。

出現在 main 函數之前的聲明語句 int power(int m, int n); 表明 power 函數有兩個 int 類型的參數,並返回一個 int 類型的值。這種聲明稱為函數原型,它必須與 power 函數的定義和用法一致。如果函數的定義、用法與函數原型不一致,將出現錯誤。

函數原型與函數聲明中參數名不要求相同。事實上,函數原型中的參數名是可選的,這樣上面的函數原型也可以寫成以下形式: int power(int, int);

但是,合適的參數名能夠起到很好的説明性作用,因此我們在函數原型中總是指明參數名。

回顧一下,ANSI C 同較早版本 C 語言之間的最大區別在於函數的聲明與定義方式的不同。按照 C 語言的最初定義,power 函數應該寫成下列形式:

/* power: raise base to n-th power; n >= 0 *//* (old-style version) */power(base, n)int base, n;{ int i, p; p = 1; for (i = 1; i <= n; ++i) p = p * base; return p;}

其中,參數名在圓括號內指定,參數類型在左花括號之前聲明。如果沒有聲明某個參數的類型,則默認為 int 類型。函數體與 ANSI C 中形式相同。

在 C 語言的最初定義中,可以在程序的開頭按照下面這種形式聲明 power 函數:int power();

函數聲明中不允許包含參數列表,這樣編譯器就無法在此時檢查 power 函數調用的合法性。事實上,power 函數在默認情況下將被假定返回 int 類型的值,因此整個函數的聲明可以全部省略。

在 ANSI C 中定義的函數原型語法中,編譯器可以很容易檢測出函數調用中參數數目和類型方面的錯誤。ANSI C 仍然支持舊式的函數聲明與定義,這樣至少可以有一個過渡階段。但我們還是強烈建議讀者:在使用新式的編譯器時,最好使用新式的函數原型聲明方式。

下面給出MFC上的實現:

void CNowaMagic_MFCDlg::OnBnClickedOk(){ // TODO: 在此添加控件通知處理程序代碼 //CDialogEx::OnOK(); //獲得EDIT CEdit* base; CEdit* n; base = (CEdit*) GetDlgItem(IDC_EDIT1); n = (CEdit*) GetDlgItem(IDC_EDIT2); CString str1; CString str2; CString showStr; char tmp[10] = ""; base -> GetWindowText(str1); n -> GetWindowText(str2); //char* pstr = (LPTSTR)LPCTSTR(str1); int my_base = _ttoi(str1); int my_n = _ttoi(str2); int result = power(my_base, my_n); showStr = itoa(result,tmp,10); CString str = _T("乘方運算結果為:"); MessageBox(str + showStr,_T("程序運行結果"),MB_OK); aseBuffer();}int power(int base, int n){ int i, p; p = 1; for (i = 1; i <= n; ++i) p = p * base; return p;}

程序運行結果:

CString轉int可以使用

int my_base = _ttoi(str1);

函數聲明注意要寫到頭函數中。

傳值調用與參數

習慣其它語言(特別是 Fortran 語言)的程序員可能會對 C 語言的函數參數傳遞方式感到陌生。在 C 語言中,所有函數參數都是“通過值”傳遞的。也就是説,傳遞給被調用函數的參數值存放在臨時變量中,而不是存放在原來的變量中。這與其它某些語言是不同的,比如,Fortran 等語言是“通過引用調用”,Pascal 則採用 var 參數的方式,在這些語言中,被調用的函數必須訪問原始參數,而不是訪問參數的本地副本。

最主要的區別在於,在 C 語言中,被調用函數不能直接修改主調函數中變量的值,而只能修改其私有的臨時副本的值。

傳值調用的利大於弊。在被調用函數中,參數可以看作是便於初始化的局部變量,因此額外使用的變量更少。這樣程序可以更緊湊簡潔。側如,下面的這個 power 函數利用了這一性質:

/* power: raise base to n-th power; n >= 0; version 2 */int power(int base, int n){ int p; for (p = 1; n > 0; --n) p = p * base; return p;}

其中,參數 n 用作臨時變量,並通過隨後執行的 for 循環語句遞減,直到其值為 0,這樣就不需要額外引入變量 i;power 函數內部對 n 的任何操作不會影響到調用函數中 n 的原始參數值。

必要時,也可以讓函數能夠修改主調函數中的變量。這種情況下,調用者需要向被調用函數提供待設置值的變量的地址(從技術角度看,地址就是指向變量的指針),而被調用函數則需要將對應的參數聲明為指針類型,並通過它間接訪問變量。

如果是數組參數,情況就有所不同了。當把數組名用作參數時,傳遞給函數的值是數組起始元素的位置或地址——它並不複製數組元素本身。在被調用函數中,可以通過數組下標訪問或修改數組元索的值。

標籤:函數 語言 學習