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ù)開(kāi)始執(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)開(kāi)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ǔ)言sizeof與字符串處理與動(dòng)態(tài)內(nèi)存分配及main函數(shù)參數(shù)詳解
- C語(yǔ)言中main函數(shù)與命令行參數(shù)詳細(xì)講解
- c語(yǔ)言中main函數(shù)用法及知識(shí)點(diǎn)總結(jié)
- C語(yǔ)言main函數(shù)的三種形式實(shí)例詳解
- c語(yǔ)言main函數(shù)使用及其參數(shù)介紹
- C語(yǔ)言main函數(shù)的參數(shù)及其返回值詳細(xì)解析
- C語(yǔ)言中怎么在main函數(shù)開(kāi)始前執(zhí)行函數(shù)
- C語(yǔ)言中main函數(shù)兩個(gè)參數(shù)的作用
相關(guān)文章
C語(yǔ)言中交換int型變量的值及轉(zhuǎn)換為字符數(shù)組的方法
這篇文章主要介紹了C語(yǔ)言中交換int型變量的值及轉(zhuǎn)換為字符數(shù)組的方法,講解了以不同進(jìn)制將整型數(shù)字轉(zhuǎn)換成字符數(shù)組,需要的朋友可以參考下2016-04-04基于Qt實(shí)現(xiàn)自定義時(shí)間選擇控件
這篇文章主要為大家詳細(xì)介紹了如何基于Qt實(shí)現(xiàn)自定義時(shí)間選擇控件,文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-12-12C語(yǔ)言編寫學(xué)生成績(jī)管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言編寫學(xué)生成績(jī)管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01C語(yǔ)言實(shí)現(xiàn)通訊錄系統(tǒng)程序
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)通訊錄系統(tǒng)程序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-06-06Qt利用QPainter實(shí)現(xiàn)基本繪圖的示例詳解
Qt?中提供了強(qiáng)大的?2D?繪圖系統(tǒng),可以使用相同的?API?在屏幕和繪圖設(shè)備上進(jìn)行繪制,它主要基于QPainter、QPaintDevice?和?QPaintEngine?這三個(gè)類。本文主要和大家介紹一下QPainter實(shí)現(xiàn)的基本繪圖,感興趣的可以了解一下2022-12-12