• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            隨筆-380  評(píng)論-37  文章-0  trackbacks-0
            對(duì)大多數(shù)不從事Linux平臺(tái)C語(yǔ)言開(kāi)發(fā)的人來(lái)說(shuō),GNU gcc的一套工具和Linux平臺(tái)的共享庫(kù)的使用還是十分陌生的,其實(shí)我也不太熟悉,姑且寫(xiě)點(diǎn)基礎(chǔ)知識(shí),權(quán)當(dāng)做備忘吧。

            一、GNU gcc的編譯工具用法

            我們先來(lái)寫(xiě)一個(gè)簡(jiǎn)單的C程序:hello.c
            C代碼 復(fù)制代碼
            1. #include?<stdio.h> ??
            2. ??
            3. void ?print_hello()?{ ??
            4. ??printf( "Hello?World\n" ); ??
            5. } ??
            6. ??
            7. int ?main( int ?argc,? char ?argv[])?{ ??
            8. ??print_hello(); ??
            9. ?? return ?0; ??
            10. }??

            定義了一個(gè)print_hello函數(shù),調(diào)用main函數(shù)打印Hello World。

            如何編譯它呢?
            C代碼 復(fù)制代碼
            1. gcc?-o?hello?-O2?hello.c??

            -o參數(shù)指定生成的可執(zhí)行程序的文件名, -O2是優(yōu)化級(jí)別。該命令會(huì)編譯生成hello可執(zhí)行程序,看看這個(gè)文件:ls -l hello
            C代碼 復(fù)制代碼
            1. -rwxr-xr-x??1?robbin?users?11939?2008-11-02?13:48?hello??

            有11KB大小。

            看看他鏈接了哪些系統(tǒng)動(dòng)態(tài)鏈接庫(kù),用ldd命令:
            C代碼 復(fù)制代碼
            1. ldd?hello??

            輸出信息為:
            C代碼 復(fù)制代碼
            1. libc.so.6?=>?/lib64/tls/libc.so.6?(0x0000002a9566d000) ??
            2. /lib64/ld-linux-x86-64.so.2?(0x0000002a95556000)??

            libc是C語(yǔ)言標(biāo)準(zhǔn)函數(shù)庫(kù),ld是動(dòng)態(tài)鏈接器。

            接著我們看看hello這個(gè)程序里面有哪些符號(hào),用nm命令:
            C代碼 復(fù)制代碼
            1. nm?hello??

            輸出:
            C代碼 復(fù)制代碼
            1. 00000000005008f8?A?__bss_start ??
            2. 000000000040043c?t?call_gmon_start ??
            3. ...... ??
            4. 00000000004004f0?T?main ??
            5. 0000000000500658?d?p.0 ??
            6. 00000000004004e0?T?print_hello ??
            7. ?????????????????U?puts@@GLIBC_2.2.5 ??
            8. 0000000000400410?T?_start??

            中間省略了一些,不過(guò)我們還是可以在符號(hào)表里面找到函數(shù)定義。

            hello有11KB,體積偏大,去處符號(hào)表可以給它瘦身,我們用strip命令:
            C代碼 復(fù)制代碼
            1. strip?hello??

            然后再ls -l hello,輸出為:
            C代碼 復(fù)制代碼
            1. -rwxr-xr-x??1?webuser?users?4464?2008-11-02?13:56?hello??

            只有4.4KB了,瘦身效果明顯! 不過(guò)這次符號(hào)表再也看不到了,nm hello,輸出為:nm: hello: no symbols。

            最后如果我們想從可執(zhí)行程序里面提取出來(lái)一點(diǎn)什么文本信息的話(huà),還可以用strings命令:
            C代碼 復(fù)制代碼
            1. strings?hello??

            輸出信息為:
            C代碼 復(fù)制代碼
            1. /lib64/ld-linux-x86-64.so.2 ??
            2. SuSE ??
            3. libc.so.6 ??
            4. puts ??
            5. __libc_start_main ??
            6. __gmon_start__ ??
            7. GLIBC_2.2.5 ??
            8. t?fff ??
            9. Hello?World??

            友情提醒一下,如果你用Java寫(xiě)一個(gè)HelloWorld.java,編譯以后你也可以用strings窺探一番。

            二、動(dòng)態(tài)共享庫(kù)怎么使用

            這次我們把hello.c拆開(kāi)成為兩個(gè)文件:hello.c和main.c。hello.c的代碼是:
            C代碼 復(fù)制代碼
            1. #include?<stdio.h> ??
            2. ??
            3. void ?print_hello()?{ ??
            4. ??printf( "Hello?World\n" ); ??
            5. }??

            而main.c的代碼是:
            C代碼 復(fù)制代碼
            1. int ?main( int ?argc,? char ?argv[])?{ ??
            2. ??print_hello(); ??
            3. ?? return ?0; ??
            4. }??


            hello.c是我們的動(dòng)態(tài)共享庫(kù),在hello.c里面我們聲明和實(shí)現(xiàn)了各種公用的函數(shù),最后main.c可以去調(diào)用這些公用函數(shù)。首先我們要把hello.c編譯成為動(dòng)態(tài)共享庫(kù):
            C代碼 復(fù)制代碼
            1. gcc?-o?libhello.so?-O2?-fPIC?-shared?hello.c???

            -fPIC參數(shù)聲明鏈接庫(kù)的代碼段是可以共享的,-shared參數(shù)聲明編譯為共享庫(kù)。請(qǐng)注意這次我們編譯的共享庫(kù)的名字叫做libhello.so,這也是Linux共享庫(kù)的一個(gè)命名的慣例了:后綴使用so,而名稱(chēng)使用libxxxx格式。

            然后編譯main.c的時(shí)候,我們需要更多的參數(shù)讓gcc知道如何尋找共享庫(kù):
            C代碼 復(fù)制代碼
            1. gcc?-o?main?-O2?-L.?-lhello?main.c???

            -L參數(shù)指定到哪個(gè)附加路徑下面去尋找共享庫(kù),現(xiàn)在我們指定在當(dāng)前目錄下面尋找;
            -l參數(shù)指定鏈接到哪個(gè)共享庫(kù)上面,我們傳的參數(shù)hello,那么gcc就會(huì)自動(dòng)鏈接到libhello.so這個(gè)共享庫(kù)上面(注意我們上面說(shuō)的libXXXX.so命名規(guī)則);
            -I參數(shù)指定到哪個(gè)附加路徑下面去尋找h文件,這個(gè)我們沒(méi)有使用。

            最后我們成功編譯好了main,執(zhí)行一下,報(bào)錯(cuò):
            引用
            ./main: error while loading shared libraries: libhello.so: cannot open shared object file: No such file or directory


            找不到libhello.so這個(gè)共享庫(kù),怎么回事?這是因?yàn)閘ibhello.so并不在操作系統(tǒng)默認(rèn)的共享庫(kù)的路徑下面,我們可以臨時(shí)指定一下鏈接路徑:
            C代碼 復(fù)制代碼
            1. export?LD_LIBRARY_PATH=.:$LD_LIBRARY_PATH??

            這樣就成功了。我們用ldd main看一下:
            C代碼 復(fù)制代碼
            1. libhello.so?=>?./libhello.so?(0x0000002a9566d000) ??
            2. libc.so.6?=>?/lib64/tls/libc.so.6?(0x0000002a9576e000) ??
            3. /lib64/ld-linux-x86-64.so.2?(0x0000002a95556000)??

            這次main程序鏈接到了libhello.so這個(gè)共享庫(kù)上面。

            三、關(guān)于Linux的動(dòng)態(tài)共享庫(kù)的設(shè)置

            可執(zhí)行程序找不到要鏈接的動(dòng)態(tài)共享庫(kù),這是Linux上面編譯和運(yùn)行程序很容易碰到的問(wèn)題,通過(guò)上面的小例子,我們已經(jīng)大致了解共享庫(kù)的一點(diǎn)基本原理,接下來(lái)我們要探討一下怎么設(shè)置程序?qū)ふ覄?dòng)態(tài)共享庫(kù)的行為。

            Linux操作系統(tǒng)上面的動(dòng)態(tài)共享庫(kù)大致分為三類(lèi):

            1、操作系統(tǒng)級(jí)別的共享庫(kù)和基礎(chǔ)的系統(tǒng)工具庫(kù)

            比方說(shuō)libc.so, libz.so, libpthread.so等等,這些系統(tǒng)庫(kù)會(huì)被放在/lib和/usr/lib目錄下面,如果是64位操作系統(tǒng),還會(huì)有/lib64和/usr/lib64目錄。如果操作系統(tǒng)帶有圖形界面,那么還會(huì)有/usr/X11R6/lib目錄,如果是64位操作系統(tǒng),還有/usr/X11R6/lib64目錄。此外還可能有其他特定Linux版本的系統(tǒng)庫(kù)目錄。

            這些系統(tǒng)庫(kù)文件的完整和版本的正確,確保了Linux上面各種程序能夠正常的運(yùn)行。

            2、應(yīng)用程序級(jí)別的系統(tǒng)共享庫(kù)

            并非操作系統(tǒng)自帶,但是可能被很多應(yīng)用程序所共享的庫(kù),一般會(huì)被放在/usr/local/lib和/usr/local/lib64這兩個(gè)目錄下面。很多你自行編譯安裝的程序都會(huì)在編譯的時(shí)候自動(dòng)把/usr/local/lib加入gcc的-L參數(shù),而在運(yùn)行的時(shí)候自動(dòng)到/usr/local/lib下面去尋找共享庫(kù)。

            以上兩類(lèi)的動(dòng)態(tài)共享庫(kù),應(yīng)用程序會(huì)自動(dòng)尋找到他們,并不需要你額外的設(shè)置和擔(dān)心。這是為什么呢? 因?yàn)橐陨线@些目錄默認(rèn)就被加入到動(dòng)態(tài)鏈接程序的搜索路徑里面了。Linux的系統(tǒng)共享庫(kù)搜索路徑定義在/etc/ld.so.conf這個(gè)配置文件里面。這個(gè)文件的內(nèi)容格式大致如下:

            C代碼 復(fù)制代碼
            1. /usr/X11R6/lib64 ??
            2. /usr/X11R6/lib ??
            3. /usr/local/lib ??
            4. /lib64 ??
            5. /lib ??
            6. /usr/lib64 ??
            7. /usr/lib ??
            8. /usr/local/lib64 ??
            9. /usr/local/ImageMagick/lib??


            假設(shè)我們自己編譯安裝的ImageMagick圖形庫(kù)在/usr/local/ImageMagick目錄下面,并且希望其他應(yīng)用程序都可以使用ImageMagick的動(dòng)態(tài)共享庫(kù),那么我們只需要把/usr/local/ImageMagick/lib目錄加入/etc/ld.so.conf文件里面,然后執(zhí)行:ldconfig 命令即可。

            ldcofig將搜索以上所有的目錄,為共享庫(kù)建立一個(gè)緩存文件/etc/ld.so.cache。為了確認(rèn)ldconfig已經(jīng)搜索到ImageMagick的庫(kù),我們可以用上面介紹的strings命令從ld.so.cache里面抽取文本信息來(lái)檢查一下:
            C代碼 復(fù)制代碼
            1. strings?/etc/ld.so.cache?|?grep?ImageMagick???

            輸出結(jié)果為:
            C代碼 復(fù)制代碼
            1. /usr/local/ImageMagick/lib/libWand.so.10 ??
            2. /usr/local/ImageMagick/lib/libWand.so ??
            3. /usr/local/ImageMagick/lib/libMagick.so.10 ??
            4. /usr/local/ImageMagick/lib/libMagick.so ??
            5. /usr/local/ImageMagick/lib/libMagick++.so.10 ??
            6. /usr/local/ImageMagick/lib/libMagick++.so??

            已經(jīng)成功了!

            3、應(yīng)用程序獨(dú)享的動(dòng)態(tài)共享庫(kù)

            有很多共享庫(kù)只被特定的應(yīng)用程序使用,那么就沒(méi)有必要加入系統(tǒng)庫(kù)路徑,以免應(yīng)用程序的共享庫(kù)之間發(fā)生版本沖突。因此Linux還可以通過(guò)設(shè)置環(huán)境變量LD_LIBRARY_PATH來(lái)臨時(shí)指定應(yīng)用程序的共享庫(kù)搜索路徑,就像我們上面舉的那個(gè)例子一樣,我們可以在應(yīng)用程序的啟動(dòng)腳本里面預(yù)先設(shè)置LD_LIBRARY_PATH,指定本應(yīng)用程序附加的共享庫(kù)搜索路徑,從而讓?xiě)?yīng)用程序找到它。
            posted on 2010-01-21 18:22 小王 閱讀(566) 評(píng)論(0)  編輯 收藏 引用 所屬分類(lèi): linux
            国产成人综合久久精品尤物| 996久久国产精品线观看| 久久91这里精品国产2020| 国内精品久久久久国产盗摄| 久久婷婷色综合一区二区| 久久婷婷五月综合成人D啪| 国产∨亚洲V天堂无码久久久| 久久综合丝袜日本网| 热久久最新网站获取| 久久99热这里只有精品66| 久久无码AV中文出轨人妻| 久久婷婷国产麻豆91天堂| 久久久WWW免费人成精品| 漂亮人妻被黑人久久精品| 久久国产视频99电影| 浪潮AV色综合久久天堂| 亚洲午夜无码AV毛片久久| jizzjizz国产精品久久| 国产精品久久久久免费a∨| 国产午夜电影久久| 久久精品午夜一区二区福利| 亚洲另类欧美综合久久图片区| 精品久久无码中文字幕| 久久亚洲中文字幕精品一区| 国内精品久久久久影院网站| 久久精品aⅴ无码中文字字幕重口| 欧美一级久久久久久久大| 日本福利片国产午夜久久| 狠狠色丁香久久婷婷综合五月| 久久亚洲色一区二区三区| 伊人热人久久中文字幕| 99久久精品午夜一区二区| 日韩人妻无码精品久久久不卡 | 久久青青草原亚洲av无码app| 国产午夜精品久久久久九九电影 | 久久精品国产亚洲一区二区三区 | 久久影视综合亚洲| 97精品伊人久久久大香线蕉| 久久精品免费观看| 久久99国产精品成人欧美| 久久婷婷人人澡人人|