• <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>
            隨筆 - 132  文章 - 51  trackbacks - 0
            <2025年8月>
            272829303112
            3456789
            10111213141516
            17181920212223
            24252627282930
            31123456

            常用鏈接

            留言簿(7)

            隨筆分類

            隨筆檔案

            文章分類

            文章檔案

            cocos2d-x

            OGRE

            OPenGL

            搜索

            •  

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            gcc 生成 .a靜態(tài)庫(kù)和 .so動(dòng)態(tài)庫(kù) 

            我們通常把一些公用函數(shù)制作成函數(shù)庫(kù),供其它程序使用。函數(shù)庫(kù)分為靜態(tài)庫(kù)和動(dòng)態(tài)庫(kù)兩
            種。靜態(tài)庫(kù)在程序編譯時(shí)會(huì)被連接到目標(biāo)代碼中,程序運(yùn)行時(shí)將不再需要該靜態(tài)庫(kù)。動(dòng)態(tài)
            庫(kù)在程序編譯時(shí)并不會(huì)被連接到目標(biāo)代碼中,而是在程序運(yùn)行是才被載入,因此在程序運(yùn)
            行時(shí)還需要?jiǎng)討B(tài)庫(kù)存在。本文主要通過舉例來說明在Linux中如何創(chuàng)建靜態(tài)庫(kù)和動(dòng)態(tài)庫(kù),以
            及使用它們。

            在創(chuàng)建函數(shù)庫(kù)前,我們先來準(zhǔn)備舉例用的源程序,并將函數(shù)庫(kù)的源程序編譯成.o文件。


            第1步:編輯得到舉例的程序--hello.h、hello.c和main.c;

            hello.c(見程序2)是函數(shù)庫(kù)的源程序,其中包含公用函數(shù)hello,該函數(shù)將在屏幕上輸出"
            Hello XXX!"。hello.h(見程序1)為該函數(shù)庫(kù)的頭文件。main.c(見程序3)為測(cè)試庫(kù)文件的
            主程序,在主程序中調(diào)用了公用函數(shù)hello。

            程序1: hello.h

            #ifndef HELLO_H
            #define HELLO_H

            void hello(const char *name);

            #endif //HELLO_H



            程序2: hello.c

            #include <stdio.h>

            void hello(const char *name)
            {
            printf("Hello %s!\n", name);
            }

            程序3: main.c

            #include "hello.h"

            int main()
            {
            hello("everyone");
            return 0;
            }

            第2步:將hello.c編譯成.o文件;

            無論靜態(tài)庫(kù),還是動(dòng)態(tài)庫(kù),都是由.o文件創(chuàng)建的。因此,我們必須將源程序hello.c通過g
            cc先編譯成.o文件。

            在系統(tǒng)提示符下鍵入以下命令得到hello.o文件。

            # gcc -c hello.c

            #

            我們運(yùn)行l(wèi)s命令看看是否生存了hello.o文件。

            # ls

            hello.c hello.h hello.o main.c

            #


            在ls命令結(jié)果中,我們看到了hello.o文件,本步操作完成。

            下面我們先來看看如何創(chuàng)建靜態(tài)庫(kù),以及使用它。

            第3步:由.o文件創(chuàng)建靜態(tài)庫(kù);

            靜態(tài)庫(kù)文件名的命名規(guī)范是以lib為前綴,緊接著跟靜態(tài)庫(kù)名,擴(kuò)展名為.a。例如:我們將
            創(chuàng)建的靜態(tài)庫(kù)名為myhello,則靜態(tài)庫(kù)文件名就是libmyhello.a。在創(chuàng)建和使用靜態(tài)庫(kù)時(shí),
            需要注意這點(diǎn)。創(chuàng)建靜態(tài)庫(kù)用ar命令。

            在系統(tǒng)提示符下鍵入以下命令將創(chuàng)建靜態(tài)庫(kù)文件libmyhello.a。

            # ar -crv libmyhello.a hello.o

            #

            我們同樣運(yùn)行l(wèi)s命令查看結(jié)果:

            # ls

            hello.c hello.h hello.o libmyhello.a main.c

            #

            ls命令結(jié)果中有l(wèi)ibmyhello.a。

            第4步:在程序中使用靜態(tài)庫(kù);

            靜態(tài)庫(kù)制作完了,如何使用它內(nèi)部的函數(shù)呢?只需要在使用到這些公用函數(shù)的源程序中包
            含這些公用函數(shù)的原型聲明,然后在用gcc命令生成目標(biāo)文件時(shí)指明靜態(tài)庫(kù)名,gcc將會(huì)從
            靜態(tài)庫(kù)中將公用函數(shù)連接到目標(biāo)文件中。注意,gcc會(huì)在靜態(tài)庫(kù)名前加上前綴lib,然后追
            加擴(kuò)展名.a得到的靜態(tài)庫(kù)文件名來查找靜態(tài)庫(kù)文件。

            在程序3:main.c中,我們包含了靜態(tài)庫(kù)的頭文件hello.h,然后在主程序main中直接調(diào)用公
            用函數(shù)hello。下面先生成目標(biāo)程序hello,然后運(yùn)行hello程序看看結(jié)果如何。

            法一 # gcc -o hello main.c -L. –lmyhello,自定義的庫(kù)時(shí),main.c還可放在-L.和 –lmyhello之間,但是不能放在它倆之后,否則會(huì)提示myhello沒定義,但是是系統(tǒng)的庫(kù)時(shí),如g++ -o main(-L/usr/lib) -lpthread main.cpp就不出錯(cuò)。

            法二 #gcc main.c libmyhello.a -o hello

            法三:先生成main.o:gcc -c main.c ,再生成可執(zhí)行文件:gcc -o hello main.o libmyhello.a,動(dòng)態(tài)庫(kù)連接時(shí)也可以這樣做。



            # ./hello

            Hello everyone!

            #

            我們刪除靜態(tài)庫(kù)文件試試公用函數(shù)hello是否真的連接到目標(biāo)文件 hello中了。

            # rm libmyhello.a

            rm: remove regular file `libmyhello.a'? y

            # ./hello

            Hello everyone!

            #

            程序照常運(yùn)行,靜態(tài)庫(kù)中的公用函數(shù)已經(jīng)連接到目標(biāo)文件中了。

            我們繼續(xù)看看如何在Linux中創(chuàng)建動(dòng)態(tài)庫(kù)。我們還是從.o文件開始。

            第5步:由.o文件創(chuàng)建動(dòng)態(tài)庫(kù)文件;

            動(dòng)態(tài)庫(kù)文件名命名規(guī)范和靜態(tài)庫(kù)文件名命名規(guī)范類似,也是在動(dòng)態(tài)庫(kù)名增加前綴lib,但其
            文件擴(kuò)展名為.so。例如:我們將創(chuàng)建的動(dòng)態(tài)庫(kù)名為myhello,則動(dòng)態(tài)庫(kù)文件名就是libmyh
            ello.so。用gcc來創(chuàng)建動(dòng)態(tài)庫(kù)。

            在系統(tǒng)提示符下鍵入以下命令得到動(dòng)態(tài)庫(kù)文件libmyhello.so。



            # gcc -shared -fPCI -o libmyhello.so hello.o (-o不可少)

            #

            我們照樣使用ls命令看看動(dòng)態(tài)庫(kù)文件是否生成。

            # ls

            hello.c hello.h hello.o libmyhello.so main.c

            #

            第6步:在程序中使用動(dòng)態(tài)庫(kù);

            在程序中使用動(dòng)態(tài)庫(kù)和使用靜態(tài)庫(kù)完全一樣,也是在使用到這些公用函數(shù)的源程序中包含
            這些公用函數(shù)的原型聲明,然后在用gcc命令生成目標(biāo)文件時(shí)指明動(dòng)態(tài)庫(kù)名進(jìn)行編譯。我們
            先運(yùn)行g(shù)cc命令生成目標(biāo)文件,再運(yùn)行它看看結(jié)果。

            # gcc -o hello main.c -L. -lmyhello



            (或 #gcc main.c libmyhello.so -o hello 不會(huì)出錯(cuò)(沒有l(wèi)ibmyhello.so的話,會(huì)出錯(cuò)),但是接下來./hello 會(huì)提示出錯(cuò),因?yàn)殡m然連接時(shí)用的是當(dāng)前目錄的動(dòng)態(tài)庫(kù),但是運(yùn)行時(shí),是到/usr/lib中找?guī)煳募模瑢⑽募ibmyhello.so復(fù)制到目錄/usr/lib中就OK了)

            # ./hello

            ./hello: error while loading shared libraries: libmyhello.so: cannot open shar
            ed object file: No such file or directory

            #

            哦!出錯(cuò)了。快看看錯(cuò)誤提示,原來是找不到動(dòng)態(tài)庫(kù)文件libmyhello.so。程序在運(yùn)行時(shí),
            會(huì)在/usr/lib和/lib等目錄中查找需要的動(dòng)態(tài)庫(kù)文件。若找到,則載入動(dòng)態(tài)庫(kù),否則將提
            示類似上述錯(cuò)誤而終止程序運(yùn)行。我們將文件libmyhello.so復(fù)制到目錄/usr/lib中,再試
            試。

            # mv libmyhello.so /usr/lib

            # ./hello

            Hello everyone!

            #

            成功了。這也進(jìn)一步說明了動(dòng)態(tài)庫(kù)在程序運(yùn)行時(shí)是需要的。

            我們回過頭看看,發(fā)現(xiàn)使用靜態(tài)庫(kù)和使用動(dòng)態(tài)庫(kù)編譯成目標(biāo)程序使用的gcc命令完全一樣,
            那當(dāng)靜態(tài)庫(kù)和動(dòng)態(tài)庫(kù)同名時(shí),gcc命令會(huì)使用哪個(gè)庫(kù)文件呢?抱著對(duì)問題必究到底的心情,
            來試試看。

            先刪除除.c和.h外的所有文件,恢復(fù)成我們剛剛編輯完舉例程序狀態(tài)。

            # rm -f hello hello.o /usr/lib/libmyhello.so

            # ls

            hello.c hello.h main.c

            #

            在來創(chuàng)建靜態(tài)庫(kù)文件libmyhello.a和動(dòng)態(tài)庫(kù)文件libmyhello.so。

            # gcc -c hello.c

            # ar -cr libmyhello.a hello.o (或-cvr )

            # gcc -shared -fPCI -o libmyhello.so hello.o

            # ls

            hello.c hello.h hello.o libmyhello.a libmyhello.so main.c

            #

            通過上述最后一條ls命令,可以發(fā)現(xiàn)靜態(tài)庫(kù)文件libmyhello.a和動(dòng)態(tài)庫(kù)文件libmyhello.s
            o都已經(jīng)生成,并都在當(dāng)前目錄中。然后,我們運(yùn)行g(shù)cc命令來使用函數(shù)庫(kù)myhello生成目標(biāo)
            文件hello,并運(yùn)行程序 hello。

            # gcc -o hello main.c -L. –lmyhello (動(dòng)態(tài)庫(kù)和靜態(tài)庫(kù)同時(shí)存在時(shí),優(yōu)先使用動(dòng)態(tài)庫(kù), 當(dāng)然,直接#gcc main.c libmyhello.a -o hello的話,就是指定為靜態(tài)庫(kù)了)

            # ./hello

            ./hello: error while loading shared libraries: libmyhello.so: cannot open shar
            ed object file: No such file or directory

            #

            從程序hello運(yùn)行的結(jié)果中很容易知道,當(dāng)靜態(tài)庫(kù)和動(dòng)態(tài)庫(kù)同名時(shí),gcc命令將優(yōu)先使用動(dòng)態(tài)庫(kù),默認(rèn)去連/usr/lib和/lib等目錄中的動(dòng)態(tài)庫(kù),將文件libmyhello.so復(fù)制到目錄/usr/lib中即可。

            Note:
            編譯參數(shù)解析
            最主要的是GCC命令行的一個(gè)選項(xiàng):
            -shared 該選項(xiàng)指定生成動(dòng)態(tài)連接庫(kù)(讓連接器生成T類型的導(dǎo)出符號(hào)表,有時(shí)候也生成弱連接W類型的導(dǎo)出符號(hào)),不用該標(biāo)志外部程序無法連接。相當(dāng)于一個(gè)可執(zhí)行文件
            -fPIC 表示編譯為位置獨(dú)立的代碼,不用此選項(xiàng)的話編譯后的代碼是位置相關(guān)的所以動(dòng)態(tài)載入時(shí)是通過代碼拷貝的方式來滿足不同進(jìn)程的需要,而不能達(dá)到真正代碼段共享的目的。

            -L. 表示要連接的庫(kù)在當(dāng)前目錄中;(多個(gè)庫(kù):在編譯命令行中,將使用的靜態(tài)庫(kù)文件放在源文件后面就可以了。比如:gcc -L/usr/lib myprop.c libtest.a libX11.a libpthread.a -o myprop
            其中-L/usr/lib指定庫(kù)文件的查找路徑。編譯器默認(rèn)在當(dāng)前目錄下先查找指定的庫(kù)文件,如前面的“法二 #gccmain.c libmyhello.a-o hello”)


            -lmyhello 編譯器查找動(dòng)態(tài)連接庫(kù)時(shí)有隱含的命名規(guī)則,即在給出的名字前面加上lib,后面加上.so或.a來確定庫(kù)的名稱libmyhello.so或libmyhello.a。
            LD_LIBRARY_PATH這個(gè)環(huán)境變量指示動(dòng)態(tài)連接器可以裝載動(dòng)態(tài)庫(kù)的路徑。
            當(dāng)然如果有root權(quán)限的話,可以修改/etc/ld.so.conf文件,然后調(diào)用 /sbin/ldconfig來達(dá)到同樣的目的,不過如果沒有root權(quán)限,那么只能采用輸出LD_LIBRARY_PATH的方法了。

            調(diào)用動(dòng)態(tài)庫(kù)的時(shí)候有幾個(gè)問題會(huì)經(jīng)常碰到,有時(shí),明明已經(jīng)將庫(kù)的頭文件所在目錄 通過 “-I” include進(jìn)來了,庫(kù)所在文件通過 “-L”參數(shù)引導(dǎo),并指定了“-l”的庫(kù)名,但通過ldd命令察看時(shí),就是死活找不到你指定鏈接的so文件,這時(shí)你要作的就是通過修改 LD_LIBRARY_PATH或者/etc/ld.so.conf文件來指定動(dòng)態(tài)庫(kù)的目錄。通常這樣做就可以解決庫(kù)無法鏈接的問題了。

            另:

            從上述可知,如何找到生成的動(dòng)態(tài)庫(kù)有3種方式:

            (1)把庫(kù)拷貝到/usr/lib和/lib目錄下。

            (2)在LD_LIBRARY_PATH環(huán)境變量中加上庫(kù)所在路徑。

            例如動(dòng)態(tài)庫(kù)libhello.so在/home/example/lib目錄下:

            $export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/example/lib

            (3) 修改/etc/ld.so.conf文件,把庫(kù)所在的路徑加到文件末尾,并執(zhí)行l(wèi)dconfig刷新。這樣,加入的目錄下的所有庫(kù)文件都可見。

            附:像下面這樣指定路徑去連接系統(tǒng)的靜態(tài)庫(kù),會(huì)報(bào)錯(cuò)說要連接的庫(kù)找不到:

            g++ -o main main.cpp -L/usr/lib libpthread.a 

            必須這樣g++ -o main main.cpp -L/usr/lib -lpthread才正確 。

            自定義的庫(kù)考到/usr/lib 下時(shí),

            g++ -o main main.cpp -L/usr/lib libpthread.a libthread.a libclass.a會(huì)出錯(cuò),但是這樣g++ -o main main.cpp -L/usr/lib -lpthread -lthread -lclass就正確了。



            轉(zhuǎn)自:http://hi.baidu.com/??????/blog/item/e58ed2f142913ea7a50f525e.html
            來自: http://hi.baidu.com/jiyeqian/blog/item/d6886e22c93f5ef8d6cae27c.html
            posted on 2012-11-03 07:53 風(fēng)輕云淡 閱讀(7300) 評(píng)論(0)  編輯 收藏 引用 所屬分類: Linux
            国产免费福利体检区久久| 久久精品国产久精国产果冻传媒| 久久国产精品成人影院| 久久精品无码一区二区三区| 99久久免费国产精品| 国产精品乱码久久久久久软件| 亚洲愉拍99热成人精品热久久| 久久久久久久综合日本亚洲| 99久久国产宗和精品1上映| 日本精品久久久久中文字幕| 国产亚洲精久久久久久无码77777| 亚洲精品高清久久| 欧美一区二区三区久久综合| 天天影视色香欲综合久久| 国产亚洲精品自在久久| 欧美精品乱码99久久蜜桃| 久久国产免费| 久久99热只有频精品8| 国内精品伊人久久久影院 | 国产精品一区二区久久不卡| 久久久中文字幕日本| 青青青国产精品国产精品久久久久| 久久婷婷国产剧情内射白浆| 久久精品国产99久久丝袜| 日本精品久久久中文字幕| 国产精品一久久香蕉国产线看观看 | 国产精品va久久久久久久| 九九精品99久久久香蕉| 无码AV波多野结衣久久| 国产69精品久久久久9999APGF | 丁香久久婷婷国产午夜视频| 97久久超碰国产精品2021| 狠狠色婷婷久久一区二区三区| 无码国内精品久久人妻| 久久国产色AV免费观看| 九九久久自然熟的香蕉图片| 久久99精品国产99久久| 久久噜噜电影你懂的| 国产高潮国产高潮久久久91| 精品久久久久一区二区三区| 青春久久|