C語(yǔ)言之沒(méi)有main函數(shù)的helloworld示例
幾乎所有程序員的第一堂課都是學(xué)習(xí)helloworld程序,下面我們先來(lái)重溫一下經(jīng)典的C語(yǔ)言helloworl
/* hello.c */
#include <stdio.h>
int main()
{
printf("hello world!\n");
return 0;
}
這是一個(gè)簡(jiǎn)單得不能再單的程序,但它包含有一個(gè)程序最重要的部分,那就是我們?cè)趲缀跛写a中都能看到的main函數(shù),我們編譯成可執(zhí)行文件并查看符號(hào)表,過(guò)濾出里面的函數(shù)如下(為了方便查看我手動(dòng)調(diào)整了grep的輸出的格式,所以和你的輸出格式是不一樣的)
$ gcc hello.c -o hello
$ readelf -s hello | grep FUNC
Num: Value Size Type Bind Vis Ndx Name
27: 000000000040040c 0 FUNC LOCAL DEFAULT 13 call_gmon_start
32: 0000000000400430 0 FUNC LOCAL DEFAULT 13 __do_global_dtors_aux
35: 00000000004004a0 0 FUNC LOCAL DEFAULT 13 frame_dummy
40: 0000000000400580 0 FUNC LOCAL DEFAULT 13 __do_global_ctors_aux
47: 00000000004004e0 2 FUNC GLOBAL DEFAULT 13 __libc_csu_fini
48: 00000000004003e0 0 FUNC GLOBAL DEFAULT 13 _start
51: 0000000000000000 0 FUNC GLOBAL DEFAULT UND puts@@GLIBC_2.2.5
52: 00000000004005b8 0 FUNC GLOBAL DEFAULT 14 _fini
53: 0000000000000000 0 FUNC GLOBAL DEFAULT UND __libc_start_main@@GLIBC_
58: 00000000004004f0 137 FUNC GLOBAL DEFAULT 13 __libc_csu_init
62: 00000000004004c4 21 FUNC GLOBAL DEFAULT 13 main
63: 0000000000400390 0 FUNC GLOBAL DEFAULT 11 _init
大家都知道用戶的代碼是從main函數(shù)開始執(zhí)行的,雖然我們只寫了一個(gè)main函數(shù),但從上面的函數(shù)表可以看到還有其它很多函數(shù),比如_start函數(shù)。實(shí)際上程序真正的入口并不是main函數(shù),我們以下面命令對(duì)hello.c代碼進(jìn)行編譯
$ gcc hello.c -nostdlib
/usr/bin/ld: warning: cannot find entry symbol _start; defaulting to 0000000000400144
-nostdlib命令是指不鏈接標(biāo)準(zhǔn)庫(kù),報(bào)錯(cuò)說(shuō)找不到entry symbol _start,這里是說(shuō)找不到入口符號(hào)_start,也就是說(shuō)程序的真正入口是_start函數(shù)
實(shí)際上main函數(shù)只是用戶代碼的入口,它會(huì)由系統(tǒng)庫(kù)去調(diào)用,在main函數(shù)之前,系統(tǒng)庫(kù)會(huì)做一些初始化工作,比如分配全局變量的內(nèi)存,初始化堆、線程等,當(dāng)main函數(shù)執(zhí)行完后,會(huì)通過(guò)exit()函數(shù)做一些清理工作,用戶可以自己實(shí)現(xiàn)_start函數(shù)
/* hello_start.c */
#include <stdio.h>
#include <stdlib.h>
_start(void)
{
printf("hello world!\n");
exit(0);
}
執(zhí)行如下編譯命令并運(yùn)行
$ gcc hello_start.c -nostartfiles -o hello_start
$ ./hello_start
hello world!
這里的-nostartfiles的功能是Do not use the standard system startup files when linking,也就是不使用標(biāo)準(zhǔn)的startup files,但是還是會(huì)鏈接系統(tǒng)庫(kù),所以程序還是可以執(zhí)行的。同樣我們查看符號(hào)表
$ readelf -s hello_start | grep FUNC
Num: Value Size Type Bind Vis Ndx Name
20: 0000000000400350 24 FUNC GLOBAL DEFAULT 10 _start
21: 0000000000000000 0 FUNC GLOBAL DEFAULT UND puts@@GLIBC_2.2.5
22: 0000000000000000 0 FUNC GLOBAL DEFAULT UND exit@@GLIBC_2.2.5
現(xiàn)在就只剩下三個(gè)函數(shù)了,并且都是我們自己實(shí)現(xiàn)的,其中printf由于只有一個(gè)參數(shù)會(huì)被編譯器優(yōu)化為puts函數(shù),在編譯時(shí)加-fno-builtin選項(xiàng)可以關(guān)掉優(yōu)化
如果我們?cè)赺start函數(shù)中去掉exit(0)語(yǔ)句,程序執(zhí)行會(huì)出core,這是因?yàn)開start函數(shù)執(zhí)行完程序就結(jié)束了,而我們自己實(shí)現(xiàn)的_start里面沒(méi)有調(diào)用exit()去清理內(nèi)存
好不容易去掉了main函數(shù),這時(shí)又發(fā)現(xiàn)必須得有一個(gè)_start函數(shù),是不是讓人很煩,其實(shí)_start函數(shù)只是一個(gè)默認(rèn)入口,我們是可以指定入口的
/* hello_nomain.c */
#include <stdio.h>
#include <stdlib.h>
int nomain()
{
printf("hello world!\n");
exit(0);
}
采用如下命令編譯
$ gcc hello_nomain.c -nostartfiles -e nomain -o hello_nomain
其中-e選項(xiàng)可以指定程序入口符號(hào),查看符號(hào)表如下
$ readelf -s hello_nomain | grep FUNC
Num: Value Size Type Bind Vis Ndx Name
20: 0000000000000000 0 FUNC GLOBAL DEFAULT UND puts@@GLIBC_2.2.5
21: 0000000000000000 0 FUNC GLOBAL DEFAULT UND exit@@GLIBC_2.2.5
22: 0000000000400350 24 FUNC GLOBAL DEFAULT 10 nomain
對(duì)比hello_start的符號(hào)表發(fā)現(xiàn)只是將_start換成了nomain
到這里我們就很清楚了,程序默認(rèn)的入口是標(biāo)準(zhǔn)庫(kù)里的_start函數(shù),它會(huì)做一些初始化工作,調(diào)用用戶的main函數(shù),最后再做一些清理工作,我們可以自己寫_start函數(shù)來(lái)覆蓋標(biāo)準(zhǔn)庫(kù)里的_start,甚至可以自己指定程序的入口
上一篇:C語(yǔ)言計(jì)算代碼執(zhí)行所耗CPU時(shí)鐘周期
欄 目:C語(yǔ)言
本文標(biāo)題:C語(yǔ)言之沒(méi)有main函數(shù)的helloworld示例
本文地址:http://www.jygsgssxh.com/a1/Cyuyan/3127.html
您可能感興趣的文章
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用函數(shù)刪除字符
- 04-02c語(yǔ)言的正則匹配函數(shù) c語(yǔ)言正則表達(dá)式函數(shù)庫(kù)
- 04-02func函數(shù)+在C語(yǔ)言 func函數(shù)在c語(yǔ)言中
- 04-02c語(yǔ)言中對(duì)數(shù)函數(shù)的表達(dá)式 c語(yǔ)言中對(duì)數(shù)怎么表達(dá)
- 04-02c語(yǔ)言用函數(shù)寫分段 用c語(yǔ)言表示分段函數(shù)
- 04-02c語(yǔ)言編寫函數(shù)冒泡排序 c語(yǔ)言冒泡排序法函數(shù)
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 04-02c語(yǔ)言分段函數(shù)怎么求 用c語(yǔ)言求分段函數(shù)
- 04-02C語(yǔ)言中怎么打出三角函數(shù) c語(yǔ)言中怎么打出三角函數(shù)的值
- 04-02c語(yǔ)言調(diào)用函數(shù)求fibo C語(yǔ)言調(diào)用函數(shù)求階乘


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語(yǔ)言函數(shù)調(diào)用后清空內(nèi)存 c語(yǔ)言調(diào)用
- 04-02func函數(shù)+在C語(yǔ)言 func函數(shù)在c語(yǔ)言中
- 04-02c語(yǔ)言的正則匹配函數(shù) c語(yǔ)言正則表達(dá)
- 04-02c語(yǔ)言用函數(shù)寫分段 用c語(yǔ)言表示分段
- 04-02c語(yǔ)言中對(duì)數(shù)函數(shù)的表達(dá)式 c語(yǔ)言中對(duì)
- 04-02c語(yǔ)言編寫函數(shù)冒泡排序 c語(yǔ)言冒泡排
- 04-02c語(yǔ)言沒(méi)有round函數(shù) round c語(yǔ)言
- 04-02c語(yǔ)言分段函數(shù)怎么求 用c語(yǔ)言求分段
- 04-02C語(yǔ)言中怎么打出三角函數(shù) c語(yǔ)言中怎
- 04-02c語(yǔ)言調(diào)用函數(shù)求fibo C語(yǔ)言調(diào)用函數(shù)求
隨機(jī)閱讀
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 04-02jquery與jsp,用jquery
- 01-10delphi制作wav文件的方法
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-10C#中split用法實(shí)例總結(jié)


