• <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>

            庫從本質上來說是一種可執行代碼的二進制格式,可以被載入內存中執行。庫分靜態庫和動態庫兩種。 

            靜態庫和動態庫的區別

            1. 靜態函數庫

                這類庫的名字一般是libxxx.a;利用靜態函數庫編譯成的文件比較大,因為整個 函數庫的所有數據都會被整合進目標代碼中,他的優點就顯而易見了,即編譯后的執行程序不需要外部的函數庫支持,因為所有使用的函數都已經被編譯進去了。當然這也會成為他的缺點,因為如果靜態函數庫改變了,那么你的程序必須重新編譯。

            2. 動態函數庫

                這類庫的名字一般是libxxx.so;相對于靜態函數庫,動態函數庫在編譯的時候 并沒有被編譯進目標代碼中,你的程序執行到相關函數時才調用該函數庫里的相應函數,因此動態函數庫所產生的可執行文件比較小。由于函數庫沒有被整合進你的程序,而是程序運行時動態的申請并調用,所以程序的運行環境中必須提供相應的庫。動態函數庫的改變并不影響你的程序,所以動態函數庫的升級比較方便。 
            linux系統有幾個重要的目錄存放相應的函數庫,如/lib /usr/lib。

            靜態庫的使用

            靜態庫的操作工具:gcc和ar 命令。 

            編寫及使用靜態庫 

            (1)設計庫源碼 pr1.c 和 pr2.c 

            [root@billstone make_lib]# cat pr1.c 

            void print1() 

                            printf("This is the first lib src!\n"); 

            [root@billstone make_lib]# cat pr2.c 

            void print2() 

                            printf("This is the second src lib!\n"); 

                   (2)  編譯.c 文件 

            [bill@billstone make_lib]$ cc -O -c pr1.c pr2.c 

            [bill@billstone make_lib]$ ls -l pr*.o 

            -rw-rw-r--        1 bill          bill                    804    4 月  15 11:11 pr1.o 

            -rw-rw-r--        1 bill          bill                    804    4 月  15 11:11 pr2.o 

            (3)  鏈接靜態庫 

                為了在編譯程序中正確找到庫文件,靜態庫必須按照 lib[name].a 的規則命名,如下例中[name]=pr. 

            [bill@billstone make_lib]$ ar -rsv libpr.a pr1.o pr2.o 

            a - pr1.o 

            a - pr2.o 

            [bill@billstone make_lib]$ ls -l *.a 

            -rw-rw-r--        1 bill          bill                  1822    4 月  15 11:12 libpr.a 

            [bill@billstone make_lib]$ ar -t libpr.a 

            pr1.o 

            pr2.o 

            (4)  調用庫函數代碼 main.c 

            [bill@billstone make_lib]$ cat main.c 

            int main() 

                            print1(); 

                            print2(); 

                            return 0; 

            (5)  編譯鏈接選項 

                -L 及-l 參數放在后面.其中,-L 加載庫文件路徑,-l 指明庫文件名字. 

            [bill@billstone make_lib]$ gcc -o main main.c -L./ -lpr 

            [bill@billstone make_lib]$ ls -l main* 

            -rwxrwxr-x        1 bill          bill                11805    4 月  15 11:17 main 

            -rw-rw-r--        1 bill          bill                      50    4 月  15 11:15 main.c 

            (6)執行目標程序 

            [bill@billstone make_lib]$ ./main 

            This is the first lib src! 

            This is the second src lib! 

            [bill@billstone make_lib]$ 

            動態庫的使用

            編寫動態庫 

            (1)設計庫代碼 

            [bill@billstone make_lib]$ cat pr1.c 

            int p = 2; 

            void print(){ 

                            printf("This is the first dll src!\n"); 

            [bill@billstone make_lib]$   

            (2)生成動態庫 

            [bill@billstone make_lib]$ gcc -O -fpic -shared -o dl.so pr1.c 

            [bill@billstone make_lib]$ ls -l *.so 

            -rwxrwxr-x        1 bill          bill                  6592    4 月  15 15:19 dl.so 

            [bill@billstone make_lib]$ 

            動態庫的隱式調用 

              在編譯調用庫函數代碼時指明動態庫的位置及名字,  看下面實例 

            [bill@billstone make_lib]$ cat main.c 

            int main() 

                            print(); 

                            return 0; 

            [bill@billstone make_lib]$ gcc -o tdl main.c ./dl.so 

            [bill@billstone make_lib]$ ./tdl 

            This is the first dll src! 

            [bill@billstone make_lib]$ 

            當動態庫的位置活名字發生改變時,  程序將無法正常運行;  而動態庫取代靜態庫的好處之一則是通過更新動態庫而隨時升級庫的內容. 

            動態庫的顯式調用 

              顯式調用動態庫需要四個函數的支持,  函數 dlopen 打開動態庫,  函數 dlsym 獲取動態庫中對象基址,  函數 dlerror 獲取顯式動態庫操作中的錯誤信息,  函數 doclose 關閉動態庫.

            [bill@billstone make_lib]$ cat main.c 

            #include <dlfcn.h> 

            int main() 

                            void *pHandle; 

                            void (*pFunc)();                                                    //  指向函數的指針 

                            int *p; 

                            pHandle = dlopen("./d1.so", RTLD_NOW);                  //  打開動態庫 

                            if(!pHandle){ 

                                            printf("Can't find d1.so \n"); 

                                            exit(1); 

                            } 

                            pFunc = (void (*)())dlsym(pHandle, "print");                //  獲取庫函數 print 的地址 

                            if(pFunc) 

                                            pFunc(); 

                            else 

                                            printf("Can't find function print\n"); 

                            p = (int *)dlsym(pHandle, "p");                                      //  獲取庫變量 p 的地址 

                            if(p) 

                                            printf("p = %d\n", *p); 

                            else 

                                            printf("Can't find int p\n"); 

                            dlclose(pHandle);                                                                //  關閉動態庫 

                            return 0; 

            [bill@billstone make_lib]$ gcc -o tds main.c –ld1 –L.

            此時還不能立即./tds,因為在動態函數庫使用時,會查找/usr/lib、/lib目錄下的動態函數庫,而此時我們生成的庫不在里邊。 這個時候有好幾種方法可以讓他成功運行: 最直接最簡單的方法就是把libstr_out.so拉到/usr/lib或/lib中去。 還有一種方法 export LD_LIBRARY_PATH=$(pwd) 另外還可以在/etc/ld.so.conf文件里加入我們生成的庫的目錄,然后/sbin/ldconfig。 /etc/ld.so.conf是非常重要的一個目錄,里面存放的是鏈接器和加載器搜索共享庫時要檢查的目錄,默認是從/usr/lib /lib中讀取的,所以想要順利運行,我們也可以把我們庫的目錄加入到這個文件中并執行/sbin/ldconfig 。另外還有個文件需要了解/etc/ld.so.cache,里面保存了常用的動態函數庫,且會先把他們加載到內存中,因為內存的訪問速度遠遠大于硬盤的訪問速度,這樣可以提高軟件加載動態函數庫的速度了。

            庫依賴的查看

            使用ldd命令來查看執行文件依賴于哪些庫。

            該命令用于判斷某個可執行的 binary 檔案含有什么動態函式庫。
            [root@test root]# ldd [-vdr] [filename]
            參數說明:
            --version  打印ldd的版本號
            -v --verbose  打印所有信息,例如包括符號的版本信息
            -d --data-relocs  執行符號重部署,并報告缺少的目標對象(只對ELF格式適用)
            -r --function-relocs  對目標對象和函數執行重新部署,并報告缺少的目標對象和函數(只對ELF格式適用)
            --help 用法信息。

            如果命令行中給定的庫名字包含'/',這個程序的libc5版本將使用它作為庫名字;否則它將在標準位置搜索庫。運行一個當前目錄下的共享庫,加前綴"./"。

            @import url(http://www.shnenglu.com/CuteSoft_Client/CuteEditor/Load.ashx?type=style&file=SyntaxHighlighter.css);@import url(/css/cuteeditor.css);
            posted on 2011-10-17 15:51 大寶天天見 閱讀(614) 評論(0)  編輯 收藏 引用 所屬分類: 7.Linux
            亚洲精品无码久久久久AV麻豆| 久久久久亚洲AV无码专区网站| 久久久久久精品久久久久| 亚洲伊人久久精品影院| 国产AV影片久久久久久| 久久精品中文无码资源站| 伊人久久大香线蕉AV一区二区| 精品国产乱码久久久久软件| 久久久久久久99精品免费观看| 狠狠色丁香久久婷婷综合_中 | 人妻无码中文久久久久专区 | 超级碰碰碰碰97久久久久| 久久久久99精品成人片直播| 久久精品亚洲福利| 久久久免费观成人影院| 人妻少妇久久中文字幕一区二区| 久久久久亚洲精品天堂久久久久久| 久久夜色精品国产噜噜麻豆| segui久久国产精品| 无遮挡粉嫩小泬久久久久久久| 国内精品久久久久影院薰衣草| 久久99精品久久久久久9蜜桃| 狠狠精品久久久无码中文字幕| 热re99久久6国产精品免费| 色99久久久久高潮综合影院| 欧美亚洲国产精品久久高清| 久久精品国产99久久久香蕉| .精品久久久麻豆国产精品| 国产无套内射久久久国产| 99精品久久精品一区二区| 久久青青草原精品国产| 久久综合亚洲色一区二区三区| 国产日韩久久久精品影院首页| 久久99热狠狠色精品一区| 久久99精品久久久久久水蜜桃| 久久99精品久久久久久| 久久精品国产久精国产思思| 久久精品国产99久久无毒不卡| 久久久亚洲欧洲日产国码二区 | 久久伊人精品一区二区三区| 久久婷婷五月综合97色直播|