C語言編程中函數的基本學習教程
C 語言中的函數等價于 Fortran 語言中的子程序或函數,也等價于 Pascal 語言中的過程或函數。函數為計算的封裝提供了一種簡便的方法,此后使用函數時不需要考慮它是如何實現的。使用設計正確的函數,程序員無需考慮功能是如何實現的,而只需知道它具有哪些功能就夠了。在 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)的定義及調用它的主程序,這樣我們可以看到一個完整的程序結構。
#include <stdio.h>
int power(int m, int n);
/* test power function */
main()
{
int i;
for (i = 0; i < 10; ++i)
printf("%d %d %d\n", 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 個或多個參數聲明)
{
聲明部分
語句序列
}
函數定義可以以任意次序出現在一個源文件或多個源文件中,但同一函數不能分割存放在多個文件中。如果源程序分散在多個文件中,那么,在編譯和加載時,就需要做更多的工作,但這是操作系統(tǒng)的原因,并不是語言的屬性決定的。我們暫且假定將 main 和 power 這兩個函數放在同一文件中,這樣前面所學的有關運行 C 語言程序的知識仍然有效。
main 函數在下列語句中調用了兩次 power 函數:printf("%d %d %d\n", i, power(2, i), power(-i, 3)); 每次調用時,main 函數向 power 函數傳遞兩個參數;在調用執(zhí)行完成時,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 本身也是函數,因此也可以向其調用者返回一個值,該調用者實際上就是程序的執(zhí)行環(huán)境。一般來說,返回值為 0 表示正常終止,返回值為非 0 表示出現異常情況或出錯結束條件。為簡潔起見,前面的 main 函數都省略了 return 語句,但我們將在以后的 main 函數中包含 return 語句,以提醒大家注意,程序還要向其執(zhí)行環(huán)境返回狀態(tài)。
出現在 main 函數之前的聲明語句 int power(int m, int n); 表明 power 函數有兩個 int 類型的參數,并返回一個 int 類型的值。這種聲明稱為函數原型,它必須與 power 函數的定義和用法一致。如果函數的定義、用法與函數原型不一致,將出現錯誤。
函數原型與函數聲明中參數名不要求相同。事實上,函數原型中的參數名是可選的,這樣上面的函數原型也可以寫成以下形式: int power(int, int);
但是,合適的參數名能夠起到很好的說明性作用,因此我們在函數原型中總是指明參數名。
回顧一下,ANSI C 同較早版本 C 語言之間的最大區(qū)別在于函數的聲明與定義方式的不同。按照 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);
str.ReleaseBuffer();
}
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 參數的方式,在這些語言中,被調用的函數必須訪問原始參數,而不是訪問參數的本地副本。
最主要的區(qū)別在于,在 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 用作臨時變量,并通過隨后執(zhí)行的 for 循環(huán)語句遞減,直到其值為 0,這樣就不需要額外引入變量 i;power 函數內部對 n 的任何操作不會影響到調用函數中 n 的原始參數值。
必要時,也可以讓函數能夠修改主調函數中的變量。這種情況下,調用者需要向被調用函數提供待設置值的變量的地址(從技術角度看,地址就是指向變量的指針),而被調用函數則需要將對應的參數聲明為指針類型,并通過它間接訪問變量。
如果是數組參數,情況就有所不同了。當把數組名用作參數時,傳遞給函數的值是數組起始元素的位置或地址——它并不復制數組元素本身。在被調用函數中,可以通過數組下標訪問或修改數組元索的值。
上一篇:C語言中static的作用及C語言中使用靜態(tài)函數有何好處
欄 目:C語言
下一篇:深入解析C++編程中線程池的使用
本文標題:C語言編程中函數的基本學習教程
本文地址:http://www.jygsgssxh.com/a1/Cyuyan/2643.html
您可能感興趣的文章
- 04-02c語言函數調用后清空內存 c語言調用函數刪除字符
- 04-02c語言的正則匹配函數 c語言正則表達式函數庫
- 04-02func函數+在C語言 func函數在c語言中
- 04-02c語言中對數函數的表達式 c語言中對數怎么表達
- 04-02c語言用函數寫分段 用c語言表示分段函數
- 04-02c語言編寫函數冒泡排序 c語言冒泡排序法函數
- 04-02c語言沒有round函數 round c語言
- 04-02c語言分段函數怎么求 用c語言求分段函數
- 04-02C語言中怎么打出三角函數 c語言中怎么打出三角函數的值
- 04-02c語言調用函數求fibo C語言調用函數求階乘


閱讀排行
本欄相關
- 04-02c語言函數調用后清空內存 c語言調用
- 04-02func函數+在C語言 func函數在c語言中
- 04-02c語言的正則匹配函數 c語言正則表達
- 04-02c語言用函數寫分段 用c語言表示分段
- 04-02c語言中對數函數的表達式 c語言中對
- 04-02c語言編寫函數冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數 round c語言
- 04-02c語言分段函數怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數 c語言中怎
- 04-02c語言調用函數求fibo C語言調用函數求
隨機閱讀
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 04-02jquery與jsp,用jquery
- 01-11ajax實現頁面的局部加載
- 08-05織夢dedecms什么時候用欄目交叉功能?
- 01-10C#中split用法實例總結
- 08-05dedecms(織夢)副欄目數量限制代碼修改
- 01-10delphi制作wav文件的方法
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10使用C語言求解撲克牌的順子及n個骰子
- 01-10SublimeText編譯C開發(fā)環(huán)境設置


