如何用C寫一個(gè)web服務(wù)器之GCC項(xiàng)目編譯
前言
本想著接下來大概實(shí)現(xiàn)一下 CGI 協(xié)議,但是實(shí)現(xiàn)過程中被一個(gè)問題卡住了:
C進(jìn)程與php進(jìn)程的交互數(shù)據(jù)類型問題:
在 C 進(jìn)程中我準(zhǔn)備將服務(wù)器處理后的請(qǐng)求數(shù)據(jù)存儲(chǔ)在一個(gè)結(jié)構(gòu)體內(nèi),然后將此結(jié)構(gòu)體中的信息傳給 PHP,而 PHP 進(jìn)程內(nèi)也會(huì)有一個(gè)全局?jǐn)?shù)組與之對(duì)應(yīng),可是眾所周之,結(jié)構(gòu)體是 C 進(jìn)程內(nèi)的內(nèi)存數(shù)據(jù),是無法直接傳給 PHP 使用的。
這時(shí)候我們也需要一種“協(xié)議”來解決進(jìn)程數(shù)據(jù)類型的異構(gòu)性。當(dāng)然這個(gè)解決方案確定起來還是很簡(jiǎn)單的,無非是對(duì)C結(jié)構(gòu)體進(jìn)行序列化,使用xml,json,protobuf(沒用過)之一,花費(fèi)時(shí)間多的地方在實(shí)現(xiàn)過程。 原來想自己造個(gè)輪子,實(shí)現(xiàn)一下json類型的編解碼,覺得有些偏離了主題了,于是考慮使用一個(gè)開源庫cJSON;
可是自己沒有過 C 大型項(xiàng)目的開發(fā)經(jīng)驗(yàn),寫的都是小 demo,gcc -o name source.c 足以解決問題了,沒有過編譯多個(gè)文件、組織項(xiàng)目的經(jīng)驗(yàn),下載到源碼后一臉懵逼,搜索到的編譯資料都是一些較為零散的內(nèi)容,不成體系,不過在自己的多次嘗試下終于成功地將 cJSON 引入到項(xiàng)目中了,這里稍做一下總結(jié)。
繞了好久,終于來到了本篇文章的主題:項(xiàng)目編譯,主要介紹一些用 GCC 在 linux 下項(xiàng)目編譯鏈接的步驟。
編譯步驟
先說一下一個(gè)C源文件的編譯一般步驟:
1.預(yù)處理(preprocess):主要是在代碼層面的處理,包括文件的引入,展開宏定義,刪除注釋,添加行號(hào)等,生成的文件以.i結(jié)尾。
gcc -E test.c -o test.i
2.編譯(compilation):編譯是在代碼語法層面的處理,生成對(duì)應(yīng)的匯編語言代碼,生成以.s
為后綴的匯編語言文件;
gcc -S test.i -o test.s
3.匯編(Assembly):將匯編語言代碼生成可執(zhí)行的機(jī)器碼,生成以.o為后綴的目標(biāo)文件。
gcc -c test.s -o test.o
4.鏈接(Linking):將各個(gè).o目標(biāo)文件連接起來,并解決庫依賴,生成無后綴的可直接執(zhí)行文件。
gcc -o test test.o
如果我們直接使用后面的命令,那么前面的步驟也會(huì)自動(dòng)執(zhí)行。如我們常使用的 gcc -o 實(shí)際上是一次性完成了所有的步驟的。
以上的中間文件,大家可以使用文本查看工具來查看其中內(nèi)容來驗(yàn)證其功能。
靜態(tài)庫和動(dòng)態(tài)庫
庫文件有動(dòng)態(tài)和靜態(tài)之分,他們的命名規(guī)范為 lib庫名.后綴,在鏈接目標(biāo)文件和庫時(shí),使用 -l 庫名(空格可省略)選項(xiàng),也可以添加-L /path來規(guī)定優(yōu)先搜索庫文件的目錄。
例如:C中的數(shù)學(xué)函數(shù)庫math.h的動(dòng)態(tài)庫文件名為libm.so,那么我們編譯連接文件時(shí)就需要添加-lm的選項(xiàng)。如果要指定庫文件路徑為/usr/lib64/libm.so,那么可添加-L /usr/lib64來指定庫文件優(yōu)先查找目錄。
另外靜態(tài)和動(dòng)態(tài)庫文件搜索目錄順序不一樣,下面分別詳細(xì)介紹:
靜態(tài)庫
靜態(tài)庫文件一般是以.a為后綴的庫文件,它在編譯連接時(shí)會(huì)將庫文件的內(nèi)容全部添加到可執(zhí)行文件中,在編譯連接完成后,靜態(tài)庫文件便不再影響可執(zhí)行文件。
它的優(yōu)點(diǎn)是簡(jiǎn)單粗暴,但如果庫文件內(nèi)部有改動(dòng)的話需要重新對(duì)所有引用此庫文件的可執(zhí)行文件重新編譯。
一般編譯步驟如下:
gcc -c static.c -o static.o // 編譯靜態(tài)庫文件的源文件 ar -r static.a static.o // 生成靜態(tài)庫文件 gcc -o main -lstatic // 連接靜態(tài)庫文件生成可執(zhí)行文件
編譯連接時(shí),靜態(tài)庫文件搜索目錄順序?yàn)椋?/p>
1.編譯連接時(shí) -L 參數(shù)指定的目錄;
2.環(huán)境變量目錄 LIBRARY_PATH;
3.固定目錄 /lib、/usr/lib、/usr/local/lib等;
動(dòng)態(tài)庫
動(dòng)態(tài)庫文件一般以.so結(jié)尾,它在編譯連接時(shí)只把動(dòng)態(tài)庫的文件添加到可執(zhí)行文件,只在程序運(yùn)行時(shí)才加載庫文件。這種方式的優(yōu)點(diǎn)是非常靈活,如果動(dòng)態(tài)庫文件內(nèi)部有變動(dòng),那么只需重要重新編譯庫文件即可。
它的一般編譯步驟如下:
gcc -c dynamic.c -fpic -o dynamic.o // 編譯動(dòng)態(tài)庫文件的源文件 -fpic 表示編譯為位置獨(dú)立的代碼,使之可以被放在可執(zhí)行文件內(nèi)存中的任何地方 gcc -shared dynamic.o -o dynamic.so // 生成動(dòng)態(tài)庫文件 gcc -o main -L . -ldynamic // 連接當(dāng)前文件夾下的動(dòng)態(tài)庫文件
編譯連接時(shí),動(dòng)態(tài)庫文件搜索目錄順序?yàn)椋?/p>
1.編譯連接時(shí) -L 參數(shù)指定目錄;
2.環(huán)境變量目錄 LD_LIBRARY_PATH;
3.配置文件/etc/ld.so.conf中配置的目錄
4.固定目錄 /lib、/usr/lib等。
CMakeLists
寫到這里還不是結(jié)尾,我們要考慮如果文件非常多怎么辦,難道每一次都要輸入n多個(gè)源文件名嗎?如果軟件完成后,用戶使用時(shí)可不想記住這些復(fù)雜的命令和文件。
自動(dòng)化才是目標(biāo),我們考慮使用自動(dòng)化編譯工具 cmake,那么接下來我們就要編寫適合項(xiàng)目文件的編譯配置文件 CMakeLists。
CMakeLists 是一個(gè) txt 文件,它就像是項(xiàng)目的編譯指南,是給用 cmake 工具用的。其語法類似于 shell,但內(nèi)置了許多函數(shù),這里我們介紹幾個(gè)簡(jiǎn)單的語法,編寫一個(gè)簡(jiǎn)單的 CMakeLists.txt。
當(dāng)前文件結(jié)構(gòu):
|__ CMakeLists.txt
|__ test.c
|__ cJSON.c
|__ include
| |__ cJSON.h
|__ lib
下面是一個(gè)動(dòng)態(tài)庫的編譯CmakeList,將解釋放在注釋中。
PROJECT(test) # 項(xiàng)目名稱 cmake_minimum_required(VERSION 2.8) # 選擇一個(gè)cmake版本 SET(LIBRARY_OUTPUT_PATH ${PROJECT_SOURCE_DIR}/lib) # 設(shè)定產(chǎn)生庫的目錄 SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_SOURCE_DIR}/bin) # 設(shè)定產(chǎn)生的可執(zhí)行文件的目錄 ADD_EXECUTABLE(test test.c) # 這里要先聲明產(chǎn)生的可執(zhí)行文件,以便后面連接 SET(cJSON cJSON.c) # 設(shè)置文件變量 ADD_LIBRARY(cJSON SHARED ${cJSON}) # 此語句用文件變量生成一個(gè)動(dòng)態(tài)鏈接庫 TARGET_LINK_LIBRARIES(test cJSON) # 連接可執(zhí)行文件與動(dòng)態(tài)鏈接庫 FIND_LIBRARY(MATH_LIB libm.so /usr/lib64) # 在/usr/lib64文件夾下找libm.so(cJSON需要) IF(MATH_LIB) TARGET_LINK_LIBRARIES(test ${MATH_LIB}) # 找到之后連接上 ENDIF() MESSAGE("cmake complete, use make to compile!") # 在命令行輸出提示語句
運(yùn)行 cmake . && make完成項(xiàng)目的構(gòu)建。
此時(shí)的目錄結(jié)構(gòu)為(略過了 cmake 產(chǎn)生的臨時(shí)文件):
|__ CMakeLists.txt
|__ test.c
|__ cJSON.c
|__ include
| |__ cJSON.h
|__ lib
| |__ libcJSON.so
|__ bin
|__ test
以上就是如何用C寫一個(gè)web服務(wù)器之GCC項(xiàng)目編譯的詳細(xì)內(nèi)容,更多關(guān)于用C寫一個(gè)web服務(wù)器之GCC項(xiàng)目編譯的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
C語言 structural body結(jié)構(gòu)體詳解用法
C 數(shù)組允許定義可存儲(chǔ)相同類型數(shù)據(jù)項(xiàng)的變量,結(jié)構(gòu)是 C 編程中另一種用戶自定義的可用的數(shù)據(jù)類型,它允許您存儲(chǔ)不同類型的數(shù)據(jù)項(xiàng),結(jié)構(gòu)用于表示一條記錄,假設(shè)您想要跟蹤圖書館中書本的動(dòng)態(tài),您可能需要跟蹤每本書的下列屬性2021-10-10C語言模擬實(shí)現(xiàn)學(xué)生學(xué)籍管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語言模擬實(shí)現(xiàn)學(xué)生學(xué)籍管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-07-07C語言實(shí)現(xiàn)掃雷游戲(可以自動(dòng)展開)
這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)掃雷游戲,可以自動(dòng)展開,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-11-11C++計(jì)算每個(gè)字符出現(xiàn)的次數(shù)
這篇文章主要介紹了C++計(jì)算每個(gè)字符出現(xiàn)的次數(shù)的相關(guān)資料,需要的朋友可以參考下2016-05-05C語言中格式化輸出符號(hào)%d、%c、%p、%x等詳解
格式化輸出在C語言中非常常用,提供了多種用法來控制輸出的格式,下面這篇文章主要給大家介紹了關(guān)于C語言中格式化輸出符號(hào)%d、%c、%p、%x等的相關(guān)資料,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-06-06