欧美极品高清xxxxhd,国产日产欧美最新,无码AV国产东京热AV无码,国产精品人与动性XXX,国产传媒亚洲综合一区二区,四库影院永久国产精品,毛片免费免费高清视频,福利所导航夜趣136

 找回密碼
 立即注冊(cè)

QQ登錄

只需一步,快速開始

搜索
查看: 3137|回復(fù): 0
打印 上一主題 下一主題
收起左側(cè)

深入了解C語言之main()函數(shù)

[復(fù)制鏈接]
跳轉(zhuǎn)到指定樓層
樓主
ID:306491 發(fā)表于 2020-4-18 13:08 | 只看該作者 回帖獎(jiǎng)勵(lì) |倒序?yàn)g覽 |閱讀模式
在各種C語言書上,能看到各式各樣main()函數(shù)的寫法,簡直令人無所適從,這是這么回事?原因主要有兩個(gè):一個(gè)是隨著C語言的發(fā)展和演化,main()函數(shù)的寫法也在不斷變化;另外,某些書籍寫法不規(guī)范或誤導(dǎo)的現(xiàn)象也同時(shí)存在。
最初main()函數(shù)的寫法非常簡潔,那個(gè)時(shí)候的C程序員哪怕一個(gè)字符似乎都不肯多寫。不知道是因?yàn)楫?dāng)時(shí)鍵盤質(zhì)量不好還是因?yàn)榫庉嬈魈愀獾木壒剩莻(gè)時(shí)代的C程序員似乎驚人地一致崇尚“簡約”——甚至可以說是“至簡”。
main()
{
       printf("hello,world\n");
}
這就是main()函數(shù)最古老的寫法,K&R在他們的經(jīng)典名著《The C Programming Language》中的第一個(gè)C語言源程序(1978)。這種寫法是那個(gè)時(shí)代的主流。
簡直和裸體差不多,連#include<stdio.h>也沒有么?在《The CProgramming Language》的第一版中確實(shí)沒有。那個(gè)時(shí)代的C語言,返回值類型為int的函數(shù)不用聲明。不過在該書的第二版(1988)中這個(gè)程序被改成了:
#include <stdio.h>
main()
{
       printf("hello,world\n");
}
返回值類型為int的函數(shù)不用聲明的規(guī)則改變了嗎?規(guī)則沒有改變。改變了的是觀念,人們已經(jīng)不再傾向于代碼的“至簡”,而開始傾向于在代碼中交代清楚每一個(gè)標(biāo)識(shí)符的來龍去脈。從C89開始倡導(dǎo)在函數(shù)調(diào)用之前一定要有函數(shù)聲明,但并沒有強(qiáng)求,而在C99這已經(jīng)是強(qiáng)制性的要求了。由于《The C Programming Language》第二版正值A(chǔ)NSI C標(biāo)準(zhǔn)頒布(1989)前夕出版,所以這種變化也應(yīng)該視為ANSI C標(biāo)準(zhǔn)的傾向性以及K&R對(duì)新標(biāo)準(zhǔn)的認(rèn)同。盡管這個(gè)例子沒有完全反映出來這種認(rèn)同。
為什么說沒有完全反映出來這種認(rèn)同呢?因?yàn)檫@個(gè)main()的定義并沒有按照函數(shù)原型(Function prototype)的方式來寫,C90中規(guī)定不帶參數(shù)的main()函數(shù)應(yīng)該這樣寫:
int main(void) { /*. . .*/}
但同時(shí)規(guī)定那個(gè)int可以省略。C90把()內(nèi)不寫任何內(nèi)容視為過時(shí)的寫法,盡管C90無奈地容忍了它(The use of function declarators with empty parentheses (notprototype-format parameter type declarators) is an obsolescent feature.)。
為什么要容忍?因?yàn)橛性S多老式的代碼還在用。
如果以C99的標(biāo)準(zhǔn)看這個(gè)main()寫得如何呢?C99不容許省略int。但同樣只把()內(nèi)不寫任何內(nèi)容視為過時(shí),而沒有完全禁止,可見習(xí)慣力量的頑固。
那又為什么說K&R對(duì)新標(biāo)準(zhǔn)的認(rèn)同呢?《The C Programming Language》第二版中的其他函數(shù)定義和函數(shù)類型聲明基本上都改成了函數(shù)原型風(fēng)格。比如,在講解main()函數(shù)的參數(shù)時(shí),K&R把原來的main()函數(shù)
#include <stdio.h>
main(argc,argv)
int argc;char *argv[];
{
       /*…… */
       return 0;
}
改成了:
#include <stdio.h>
main(int argc, char *argv[])
{
       /*…… */
       return 0;
}
前一個(gè)寫法今天已經(jīng)差不多絕跡,后一個(gè)main()以今天的眼光來看有些奇怪,main()的參數(shù)是用函數(shù)原型風(fēng)格寫的,但卻沒有寫main()返回值的類型,給人有點(diǎn)半新半舊的感覺。盡管不能說它違背C90(因?yàn)镃90容許不寫main()前面的int),但如果寫上了返回值的類型int,就同時(shí)滿足現(xiàn)代C99標(biāo)準(zhǔn)的要求了。
這里出現(xiàn)的“return 0;”是怎么回事?這在現(xiàn)代C語言中已經(jīng)是司空見慣了,它返回給操作系統(tǒng)一個(gè)值以表明程序是在何種狀態(tài)下結(jié)束的。但在另一段代碼中,K&R似乎又走得太遠(yuǎn):
#include <stdio.h>
main(int argc,char *argv[])
{
       int found = 0 ;
       /*……計(jì)算found的值 */
       return found;
}
這個(gè)實(shí)在有些“標(biāo)新立異”,居然把計(jì)算結(jié)果返回給了操作系統(tǒng),頗有突破常規(guī)之嫌。
那前面幾個(gè)沒有“return 0;”的main()函數(shù)會(huì)怎么樣?按照C90標(biāo)準(zhǔn),會(huì)返回一個(gè)不確定的int類型的值,如果確實(shí)不關(guān)心這個(gè)返回值是多少,不寫確實(shí)可以。但C99卻要求編譯器在編譯的時(shí)候幫忙給補(bǔ)上這個(gè)“return 0;”,C99在必須寫int這個(gè)問題上沒有遷就懶人,但在這里卻對(duì)偷懶的做法給予了遷就。問:如果確實(shí)不關(guān)心main()函數(shù)的返回值,把main()的返回值定義為void類型如何?我看到許多書上都這樣寫的。
#include <stdio.h>
void main()
{
       printf("This isa C program.\n");
}
這在C99之前是一種野路子寫法,究竟從哪里冒出來的,無據(jù)可考。但前幾年的主流教科書中這種寫法很常見。K&R(C語言的發(fā)明者)沒有這樣寫過,C90國際標(biāo)準(zhǔn)也不承認(rèn)這種寫法。Bjarne Stroustrup(C++語言的創(chuàng)始人)在他的關(guān)于C++的FAQ中,在回答是否可以寫“void main()”時(shí)憤怒地回答說這種寫法在C++和C中都不曾有過。事實(shí)上,很多C語言專家都認(rèn)為“void main()”非常邪惡。
因此,在C99之前,這是不符合標(biāo)準(zhǔn)的寫法。盡管這段代碼的功能似乎是輸出“This is a C program.”,但其實(shí)卻不是一個(gè)“C program”。
但是有時(shí)這樣寫并沒有產(chǎn)生錯(cuò)誤啊?首先,C語言的錯(cuò)誤不一定反應(yīng)在編譯、鏈接或運(yùn)行過程中。你輸出一個(gè)垃圾值也可能一路通過編譯、鏈接或運(yùn)行,但這不說明你的代碼沒有錯(cuò)誤,更不能說明這樣的代碼正確、有意義。其次,這樣的寫法在有些編譯器下程序會(huì)產(chǎn)生崩潰或得到警告。這說明這種寫法至少不普遍性適用的。可以說,如果不是C99標(biāo)準(zhǔn),這種寫法根本沒有立錐之地。
C99給了這種寫法以立足之地么?從某種意義上也許可以這樣理解。因?yàn)镵&R沒承認(rèn)過這種寫法,C90根本不承認(rèn)這種寫法,C99雖然沒有正式承認(rèn)這種寫法,但為這種寫法留了一個(gè)后門:“It shall be defined ……or in some other implementation-definedmanner”。這意思就是說,如果編譯器明確聲稱允許void main()這種寫法的話,那么C99不再象C90那樣簡單認(rèn)為這種寫法違背C標(biāo)準(zhǔn)。
但是不管怎么說,這種寫法最多是某些編譯器的一種“方言土語”,如果沒有特殊理由,比如僅僅是工作在某個(gè)特殊環(huán)境,且僅僅使用特定的編譯器而根本不考慮程序的可移植性,為什么不寫普遍適用的形式呢?
既然很多C語言專家都認(rèn)為“void main()”非常邪惡,C99為什么包容這種寫法呢?很難確定C99是否就是打算專門想把這種寫法也“收容”在標(biāo)準(zhǔn)之列。因?yàn)槌藇oid main(),還有另外一些main()函數(shù)的寫法被C90排除在標(biāo)準(zhǔn)之外了。而現(xiàn)在,這些寫法在理論上也具備了符合C99標(biāo)準(zhǔn)的可能性。
還有什么樣的main()函數(shù)?很多編譯器都支持下面的main()的寫法:
int main(int argc, char *argv[], char *env[])
{
       /* */
       return 0;
}
居然有3個(gè)形參,那個(gè)env是做什么用的?那個(gè)參數(shù)可以使程序獲得環(huán)境變量。
什么叫環(huán)境變量?簡單地講可以理解為操作系統(tǒng)記錄的一些數(shù)據(jù),比如計(jì)算機(jī)的名字,操作系統(tǒng)放在哪里等等。應(yīng)用程序在運(yùn)行時(shí)可能要用到這些信息,這時(shí)可以通過env這個(gè)參數(shù)來獲得。
如果編譯器不支持main()的第三個(gè)參數(shù)怎么辦?標(biāo)準(zhǔn)庫函數(shù)也可以達(dá)到同樣的目的。
#include <stdlib.h>
char *getenv(const char *name);
是否可以說void main()和int main(int argc, char *argv[], char *env[])也符合C99標(biāo)準(zhǔn)呢?恐怕還不能這么說,現(xiàn)在只是不能說這兩種寫法一定不符合C99標(biāo)準(zhǔn)。但這兩種寫法不符合C90標(biāo)準(zhǔn)是確定的,這兩種寫法的可移植性很差也是確定無疑的。C99標(biāo)準(zhǔn)在這里寫的很模糊,沒有進(jìn)一步界定“implementation-defined manner”的含義。除非編譯器聲明遵守C99標(biāo)準(zhǔn),且容許這兩種寫法,否則斷言這兩種寫法符合C99標(biāo)準(zhǔn)屬于空穴來風(fēng)。
有人說“C99建議把main函數(shù)指定為int型”,這種說法對(duì)嗎?顯然不對(duì)。因?yàn)镃99并非絕對(duì)不包容返回值非int類型的main()。正確的說法是,C90要求main()函數(shù)的返回值一定得是int型。但C90容許不寫那個(gè)int,而C99則要求必須寫上這個(gè)“int”。
下面這種風(fēng)格如何?
#include <stdio.h>
int main()
{
       printf("This isa C program.\n");
       return 0;
}
這個(gè)寫法有點(diǎn)不倫不類。返回值的類型int寫了,這個(gè)和C89的倡導(dǎo)或C99的要求一致,但是()里面什么都不寫,又與標(biāo)準(zhǔn)的所倡導(dǎo)的風(fēng)格不符,所以說不倫不類。這種寫法目前的標(biāo)準(zhǔn)依然容許,但屬于標(biāo)準(zhǔn)目前尚能容忍的但即將過時(shí)的(obsolescent)寫法,被拋棄只是早晚的問題。這種寫法就如同古代的函數(shù)形參的寫法一樣:
main(argc,argv)
int argc;char *argv[];
{
       /*…… */
       return 0;
}
都屬于歷史的垃圾。
見過在main()的函數(shù)體的“}”之前前寫一句getch();,這個(gè)是怎么回事?這個(gè)是時(shí)代的產(chǎn)物。在PC從DOS時(shí)代轉(zhuǎn)變?yōu)閃indows時(shí)代的過程中,DOS時(shí)代的IDE(主要是TC)無法在運(yùn)行程序后顯示輸出結(jié)果,為了在運(yùn)行后從容仔細(xì)地觀察一下運(yùn)行結(jié)果再返回IED界面,加上了這么一句,人為地延長程序運(yùn)行時(shí)間(因?yàn)間etch()會(huì)等待用戶輸入一個(gè)字符)。但這與main()本身的結(jié)構(gòu)無關(guān)。這條語句不具備普遍意義,只是將就過時(shí)的IDE的一種權(quán)宜之計(jì)而已。所謂不具備普遍意義是指,第一,真正的程序往往不需要這條語句,就是說這條語句與程序功能無關(guān);第二,getch()這個(gè)函數(shù)并不是標(biāo)準(zhǔn)函數(shù),只有個(gè)別的編譯器才支持它,在其他編譯器上寫這條語句,很可能行不通。
為什么不用getchar()這個(gè)標(biāo)準(zhǔn)庫函數(shù)呢?getchar()的功能和getch()有點(diǎn)區(qū)別,前者會(huì)在標(biāo)準(zhǔn)輸出設(shè)備上顯示用戶鍵入的字符,這顯得很不利索,而后者則不會(huì)顯示用戶所鍵入的字符,更接近“Press Any Key to continue……”的效果。
有的代碼在main()函數(shù)結(jié)束前寫system("PAUSE");,是否也是這個(gè)意思?是的。這也是一種人工制造的“請(qǐng)按任意鍵繼續(xù). . .”,與程序功能結(jié)構(gòu)無關(guān),只是為了方便地觀察輸出結(jié)果。但是這種寫法比調(diào)用getch()要好,因?yàn)閟ystem()函數(shù)是標(biāo)準(zhǔn)庫函數(shù),各個(gè)編譯器都提供支持。
有一種說法,“在最新的C99標(biāo)準(zhǔn)中,只有以下兩種定義方式是正確的:”
int main( void )
{
       /*  */
       return 0;
}
int main( int argc, char *argv[] )
{
       /*  */
    return 0;
}
這種說法對(duì)嗎?
這種說法顯然不對(duì)。但可以確認(rèn)的是這兩種定義方式一定正確。不但在C99來說是正確的,以C89來說也是正確的。
還有一種寫法:
int main( void )
{
       return EXIT_SUCCESS;
}
那個(gè)EXIT_SUCCESS是怎么回事?
return EXIT_SUCCESS;是與return 0;等價(jià)的一種文雅的寫法。EXIT_SUCCESS是在stdlib.h中定義了的符號(hào)常量,返回這個(gè)值表示程序任務(wù)完成后程序退出。在stdlib.h定義的另一個(gè)符號(hào)常量EXIT_FAILURE,通常用于程序無法完成任務(wù)而退出。
實(shí)在太眼花繚亂了,需要記住這么多嗎?顯然沒必要。很多東西都是歷史原因遺留下的垃圾。
如果學(xué)習(xí)C語言,應(yīng)該記住或使用哪種呢?顯然是:
int main( void )
{
       /*  */
    return 0;
}
int main( int argc, char *argv[] )
{
       /*  */
    return 0;
}
第一,他們普遍適用,不存在可移植性的問題;
第二,就目前看,他們不存在任何過時(shí)或即將過時(shí)的成分。當(dāng)然,如果喜歡文雅,不寫return 0;而寫EXIT_SUCCESS也可以。順便說一句,有的學(xué)習(xí)者記不住帶參數(shù)main()函數(shù)兩個(gè)形參的名字。其實(shí)這兩個(gè)形參的名字也可以自己取,不一定用那兩個(gè)名字,只要記住類型就可以了。第二個(gè)參數(shù)的類型也可以是char **,這和原來的是等價(jià)的。

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂 踩
回復(fù)

使用道具 舉報(bào)

本版積分規(guī)則

小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表