您好,登錄后才能下訂單哦!
今天就跟大家聊聊有關linux程序開發如何入門,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結了以下內容,希望大家根據這篇文章可以有所收獲。
總會看到有人說Linux上的應用程序開發是高手才可以完成的,而且這種“迷信”在目前似乎還很普遍。然而,情況并不是這樣的,從程序庫的支持方面,Linux平臺為用戶級應用程序的開發提供了很多功能強大且豐富的程序庫,而且它們大部分是跨平臺的(Boost、OpenGL、STL、Qt、Java等)和基于POSIX標準的(glibc等),同時Linux內核還為驅動程序的開發提供了功能完備的內核接口,從開發工具方面,Linux提供了功能強大的編譯器GCC和調試器GDB,借助它們的幫助,我們可以很輕松的在Linu x上開發出可移植性的應用程序。既然如此,“迷信”又源于何來呢?我想,一方面由于詳細介紹Linux各種開發的書籍較少,各種Linux應用在國內仍不普及,另一方面則是由于很多人在安裝好一個Linux后,苦于找不到一個得心應手的IDE環境,從而感到不知所措,畢竟,我們很多人都習慣了寫好程序后,按下F5,剩下的任務就讓IDE全權代理了。其實想在Linux下如此這般當然也沒問題。既然說到了IDE,就讓我們從它開始吧,相信選擇一個好的IDE環境是你整個學習過程的一個不錯的開始。
工欲善其事必先利其器——IDE篇
其實Linux下有許多功能強大的IDE環境,因為從某種意義上說,Linux是專為開發者準備的操作系統,這個東西當然少不了,在這里為讀者介紹一些比較常用的IDE。
KDevelop
這是一個用Qt開發的IDE,其主要支持的語言是C / C++,
Eclipse
近年來,eclipse可以說發展極為迅速,它不僅是一個以java為主的開發平臺,其功能強大的插件體系結構使得它可以被當作各種應用程序來使用。作為各種插件的載體,eclipse提供了完整的GUI接口,用戶完全可以借助eclipse來只關心自己想做的工作。
Emacs
VIM
山高月曉水落石出——IDE后臺的故事GCC篇
前面我們簡要介紹了一些IDE環境,其中所有C/C++相關程序的編譯都是由GCC來完成的,而IDE只不過起到了一個收集編譯信息和為我們的項目生成makefile等作用(后面我們會提到)。出于目前Linux開發的特點,C仍是系統開發的主流語言。所以,對GCC有一個全面的了解是很有必要的,一旦IDE不能滿足你的需求,我們要有手工打造程序的能力,而且出于學習的目的,我們往往不需要IDE生成的那些復雜的文件,為一個Hello world生成2M多的文件顯然是多余的。
GCC的全稱是GNU Compiler Collection,從這個名字我們不難看出,GCC代表著一個編譯器的集合,目前GCC可以支持C, C++, Objective-C, Objective-C++, Fortran, Java, and Ada等語言。但是出于一般性考慮,我們這里只討論GCC中的C/C++部分。
目前GCC的***發布版是4.0.0,但是這個版本由于使用了新技術和新的編碼規范,很多舊的代碼都需要修改才可以通過編譯,所以并不推薦使用這個版本。而相對穩定的新版本目前是3.4.4,大家可以到GNU的主頁上更新下載。那么究竟GCC強大在哪里,如何使用?下面我就通過幾個簡單而實際的例子帶你看看GCC提供的強大功能。
通過Helloworld的編譯熟悉GCC的基本使用方法
似乎為所有新語言提供一個Hello World樣本程序已經成為了一種不成文的標準,人們通過它來認識語言的一些基本要素。在這里,我們使用一個Hello World來看看如何用GCC生成可執行文件。
把上面的文件存成helloworld.c,之后打開控制臺,輸入如下的命令gcc helloworld.c –o helloworld如果一切正常的話,你的控制臺上應該沒有任何輸出。用ls查看你的工作目錄,你會發現目錄下多了一個名為helloworld的可執行文件,之后,執行./hellworld
就會看到這個程序的輸出了
很簡單不是嗎?但是學過計算機的朋友都應該知道,程序的編譯過程要分為下圖所示的過程而GCC的強大之處就在于它允許你在上面所示的任何一個過程中停下來查看中間結果,并對其加以控制。
1. 預處理首先是預處理過程,GCC的-E選項可以讓GCC在預處理后停止編譯,并向標準輸出打印預處理過后的文件。下面的-o用于指定輸出文件的文件名。
gcc –E hellowrold.c –o helloworld.cpp下面是helloworld.cpp的一部分的內容,我們看到,文件已經包含了stdio.h中的內容。
如果我們想執行下一步的編譯過程,可以繼續使用GCC的-x
l c c-header c-cpp-output
l c++ c++-header c++-cpp-output
l assembler assembler-with-cpp
另外,下表列出了常用的GCC后綴名
當然,你也可以省略掉language type的部分,這時候GCC會根據文件的后綴名自行判斷,就像你沒有使用該選項一樣。
下面繼續我們的編譯過程
2. 編譯如果我們想獲得編譯后的源文件可以使用-S選項,該選項讓gcc只執行編譯(生成匯編文件)而不進行匯編(生成目標文件),此時,我們可以用-o選項指定輸出的匯編文件的名稱。
gcc –S helloworld.cpp –o hellowrld.S
3. 匯編另外,我們還可以使用GCC的-c選項來編譯和匯編源文件而不鏈接,此時-o指定的輸出文件就是編譯后的目標文件名gcc –x c++ -c helloworld.cpp –o helloworld.o
4. 鏈接***,我們可以利用GCC來把我們剛才生成的.o文件鏈接成可執行程序gcc helloworld.o –o helloworld這一次,我們使用了-o選項指定了可執行文件名,也就是說,根據輸入文件類型的不同,-o有著不同的含義。
5. 函數庫的鏈接和包含文件對于我們編寫的任和一個程序,沒有庫函數的支持是不可想象的,而當我們要使用的頭文件和函數庫不在GCC默認的搜索路徑下的時候(例如OpenGL、Qt、KDE、Boost等),我們就需要手工來告訴GCC他們的位置。
先來看頭文件路徑的指定。我們可以利用-I
–L/usr/X11R6/lib–lX11
另外,GCC在默認情況下使用共享庫來鏈接程序,而當你想鏈接靜態庫的時候,一定要使用-static選項,例如-lncurses -static
在這一部分的***,我們對編譯時用到的GCC常用命令做一個簡要的總結
上面,我們提到了關于GCC編譯的常用命令,這里另外補充一些幫助性的常用命令,他們可以讓你對GCC的基本配置和使用作一個了解。
在這部分的***,我們來談一談關于構建軟件時鏈接參數的設定問題。在上面的第5部分我們已經提到了,函數庫的使用是需要-L和-l一起配合來使用的,但實際上,往往一個像樣的程序需要很多庫的支持,例如,如果你需要編寫一個GTK程序,我們需要下面的鏈接參數:-L/usr/lib -L/usr/X11R6/lib -lgtk -lgdk -rdynamic -lgmodule -lglib -ldl -lXi -lXext -lX11 –lm,看上去有些嚇人,你可能會問,我如何知道需要這些呢,如果我想編寫KDE的程序呢,還有OpenGL呢?其實,情況比你想象的要好很多,在/usr/bin目錄下,有很多名為xxx-config的腳本,它們的作用就是向用戶顯示編譯鏈接程序時使用的參數的。這些腳本可以接受一些參數,比較常用的有—libs用于列出鏈接特定程序時使用的程序庫,另外--cflags用于生成頭文件的包含目錄,也就是上面我們提到的-I參數。于是,對于GTK程序,我們可以使用下面的命令來編譯:gcc gtksource.c `gtk-config –libs --cflags`當然,為每一種程序寫一個config顯然不是一個好辦法,目前新的開發包都使用pkg-config這個腳本來生成鏈接參數。你可以使用pkg-config –list-all查看pkg-config支持的所有鏈接參數當你在上面這份列表中查到了自己想要程序包時,就可以使用下面的命令來編譯程序了gcc.suffix `pkg-config
讓GCC幫助你更好的工作
上面我們簡單介紹了GCC的常用命令行選項,其實GCC的功能比上面提到的那些要豐富得多,GCC對代碼的警告、優化、調試等方面提供了豐富的支持,下面我們就從一些例子來看看GCC提供的這些功能。
1.對問題代碼提出警告
GCC對程序代碼提供了完整的檢查功能,由于C/C++語言本身的特點,很多錯誤都是程序員無意間犯下的,例如使用了未定義的變量、在bool表達式中使用了=而不是==等等問題,利用GCC提供的代碼檢查功能,我們可以讓編譯器為我們找到這些問題,避免運行時發生災難。
首先,我們來看一個“問題代碼”
/* test_warning.c We use this file to check the warning facilities provided by GCC*/ #include
void main() { /* main should return int*/
int a, b; long long l = 2.2; /* long long type is GNU extension, not standard ANSI / ISO type*/
miss_decl(); /* We call an undeclared function*/
if (a = 0) /* May be we want == here instead of =*/
printf (“a really equals to 0?\n”); if (b != 0) /* We used uninitialized variables*/ /* %d and “We should put b here” don‘t match*/ printf(“We make a mistake again! b = %d\n”, “We should put b here”);
};
void miss_decl() {
/* /* This type of annotation is prohibited*/
printf(“We should put the declaration before it‘s been used!\n”);
}
上面這些代碼故意制造了很多編程中出現的常見問題,接下來,我們就用這段代碼來檢測一下GCC提供的各種常用的警告設施。
首先,我們不使用任何警告設施編譯上面的程序gcc test_warning.c –o test_warning默認情況下,GCC會給出輸出,其中GCC識別出了main函數不標準(warning)以及使用了未聲明的函數(error)兩個問題,但是其他的GCC并未察覺。
1.利用-pedantic找出不符合ANSI / ISO標準的代碼
執行下面的命令:gcc –pedantic test_warning.c –o test_warning可以看到,這次GCC以警告的形式報告了代碼中long long的使用,但是要說明的是我們并不能依賴這個選項來保證我們的代碼完全符合ANSI / ISO標準,因為該選項只報告ANSI C要求編譯器進行檢察的內容。另外,你還可以使用-pedantic-errors讓GCC把所有的警告都變成錯誤。
2. 利用-Wformat檢查printf中的參數不匹配問題執行下面的命令:gcc –Wformat test_warning.c –o test_warning
3. 利用-WComment找出注釋中的錯誤執行下面的命令:gcc –WComment test_warning.c –o test_warning
4. 利用-Wparentheses查找bool表達式中的=錯誤執行下面的命令:gcc –Wparentheses test_warning.c –o test_warning
5. 用-Wuninitialized查找未初始化變量的使用執行下面的命令:gcc –O –Wuninitialized test_warning.c –o test_warning值得說明的是,在使用這個選項的時候,一定要配合上-O(后面我們會提到)選項
6. 利用-Wimplicit-function-declaration / -Werror-implicit-function-declaration檢查未聲明函數的使用執行下面的命令:gcc -Wimplicit-function-declaration test_warning.c –o test_warning另外-Werror-implicit-function-declaration和-Wimplicit-function-declaration作用是類似的,只是如果你使用了未聲明的函數,前者會把它認為是一個錯誤。
7. 如果你只是想對你的代碼進行全面的檢查,你大可不必把上面的選項一并列出來,GCC提供了-Wall選項,含義就是列出所有代碼中的警告執行下面的命令:gcc –Wall test_warning.c –o test_warning 8. 如果你想走另一個極端,也就是不想讓gcc輸出任何警告,那么使用-w選項,該選項禁止所有的警告執行下面的命令:gcc –w test_warning.c –o test_warnin
<輸出結果>
對于上面所有的選項,你都可以把它們和-Werror選項一起使用,這樣就可以把所有的警告都變成錯誤。另外,如果你只是想對代碼進行檢查而并不執行編譯的話,可使用-fsyntax-only選項,像下面的命令這樣gcc –fsyntax-only test_warning.c基本上來說,我們常用的一些警告選項就是這些,而其中-Wall更是我們極為常用的功能。
2. 優化選項這一部分的內容可以分成兩部分,一部分是讓編譯器對代碼進行分析
后,進行的代碼優化,另一部分是我們可以為編譯器制定一些關于硬件的信息,讓他生成對硬件結合的更好的代碼,而我們之所以要用源代碼來編譯程序,很多情況下,是出于這方面的原因。
首先來看代碼優化,從代碼的整體優化上,GCC提供了下面的選項
-O–O1
這兩個選項的含義是一樣的,GCC將執行減少代碼尺寸和執行時間的優化,對于那些會嚴重影響編譯時間的優化選項,這個級別的優化并不會執行。
-O2
在這一級別GCC將會提供所有支持的優化,但這其中并不包括以空間換時間的優化手段,例如編譯器不會使用循環展開和函數內聯。和-O相比,該選項進一步加快了編譯時間和生成代碼的性能。
-O3
除了-O2提供的優化選項外,還指定了-finline-functions,-funswitch-loops和-fgcse-afer-reload選項,目的只有一個就是全力執行代碼優化。
-Os
這個選項是專門用來優化代碼尺寸的,-Os打開了所有-O2級別中不會顯著增長代碼尺寸的優化選項
-O0
該選項代表不執行優化
在這里要說明的是,盡管GCC提供了1~3和s這4個整體優化選項,但從實際的優化效果上來看,往往O3優化出來的程序的效率并不是***的,而大部分情況下我們都在使用-O2,如果你希望獲得***的效率利益,那么不妨這4個選項都試試。另外,其實這些選項只不過是GCC提供的很多單方面優化的一個組合,如果你想了解更為具體的優化內容,可以去查看GCC手冊,出于篇幅限制,這里不細談了。***要記住的一點是,如果你的程序是用于高精度數值計算的,那么記住不要使用上面任何的優化選項。
下面來看基于硬件優化,由于這部分和計算機硬件相關,這里僅用Intel的CPU做一些說明:對于所有為Intel和AMD x86-64提供的優化選項都是用m開頭的,下面寫一些常用的選項:
-march
該選項用來指定CPU的類型,常用的有i386 \ i486 \ i586 \ pentium-mmx \ i686 \ pentium2 \ pentium3 \ pentium-m \ pentium4 \ prescott \ k6 \ athlon \ athlon-4 \ k8等等,讀者可以根據自己的情況進行指定。
-mfpmath
該選項用于指定浮點運算單元的類型。包括
387
使用標準的數學協處理器
sse
使用SSE指令集提供的標量浮點運算。在Pentium3 \ Athlon-4以及更新的芯片上支持這個特性。另外,在pentium4以及AMD x86-64處理器上,SSE2還可以進行雙精度浮點計算。
sse,387混合使用387數學協處理器和SSE指令集,該選項可以充分的利用CPU的浮點寄存器和xmm寄存器,但是該選項還處在試驗階段。
-malign-double
該選項使得GCC把double \ long double \ long long類型的變量在4字節或2字節地址上對齊,在Pentium級的CPU上,這會使得代碼的執行速度更快,當然帶來的代價是需要更多的內存來執行程序。-mmmx –msse –msse2 –msse3 –m3dnow這些選項用來啟動內置函數直接使用這些處理器擴展指令的功能。在編譯3D或多媒體程序的時候,使用他們是非常有效的。
3. 對調試的支持當程序出錯的時候,我們可以在Visual Studio中輕松的進行調試,而在Linux中,一旦出現Segmentation Fault,似乎我們除了用眼睛去看代碼就沒有更好的選擇了,其實情況不然,用GCC向程序加入一些適當的調試信息,我們可以利用GDB去調試程序。在這里,我們介紹最為常用的-g和-ggdb選項。
先來看-g。該選項可以利用操作系統的“原生格式(native format)”生成調試信息。GDB可以直接利用這個信息。盡管我們可以把-O和-g放在一起使用,但是,這種做法是極為不推薦的。
如果你想用GDB來調試程序,那么你可以使用-ggdb來讓GCC為GDB生成更為豐富的調試信息,但是,此時你就不能用其他的調試器來進行調試了。
最后要說明的是,上面這兩個選項都可以接受一個輸出調試信息的級別,默認的級別是2。如果你指定1級(-g1),那么GCC會生成最少的調試信息,這包括函數和全局變量的描述信息,但是對于局部變量和行號等信息,在這個級別是不會輸出的。另外一個級別是3級(-g3),在這一級別上,GCC會為程序中的所有宏定義和符號生成調試信息。
看完上述內容,你們對linux程序開發如何入門有進一步的了解嗎?如果還想了解更多知識或者相關內容,請關注億速云行業資訊頻道,感謝大家的支持。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。