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

            isware

            使用Automake 創建和使用靜態庫/動態庫

            1. 目錄結構如下:
            example
            |——src 目錄(存放源代碼文件)
                    |——hello.c
            |——lib 目錄(存放用來生成庫的文件)
                    |——test.c 用來生成靜態庫libhello.a
            |——include 目錄(存放程序中使用的頭文件)
                    |——hello.h
            2. 編寫的各個目錄下的源文件
            hello.h 文件
            extern void print(char *);
            test.c 文件
            #include<stdio.h>
            void print(char *msg)
            {
            print(“%s\n”, msg);
            }
            hello.c 文件
            #include “hello.h”
            int main()
            {
            print(“Hello static library!”);//這里用到的是靜態庫中的函數
            return 0;
            }
            3. 編寫lib/Makefile.am 文件
            noinst_LIBRARIES=libhello.a
            libhello_a_SOURCES=test.c
            AUTOMAKE_OPTIONS=foreign
            第一行noinst 表示生成的是靜態庫,不需要make install ,直接制定它的位置和名字就
            可以使用。
            第二行表示用來生成靜態庫的源文件。如果要把靜態庫生成到其他地方,可以在=后面
            加上路徑(建議用絕對路徑,并將所要用到的靜態庫生成在同一個文件夾下,如lib)。
            第三行AUTOMAKE_OPTIONS 是Automake 的選項。Automake 主要是幫助開發 GNU 軟
            件的人員來維護軟件,所以在執行Automake 時,會檢查目錄下是否存在標準 GNU 軟件中
            應具備的文件,例如 'NEWS'、'AUTHOR'、 'ChangeLog' 等文件。設置為foreign 時,Automake
            會改用一般軟件的標準來檢查。如果不加這句的話,需要在autoconf之前,先執行touch NEWS
            README AUTHORS ChangeLog 來生成'NEWS'、'AUTHOR'、 'ChangeLog' 等文件
            4. 編寫src/Makefile.am 文件
            AUTOMAKE_OPTIONS=foreign
            INCLUDES= -I../include
            bin_PROGRAMS=hello
            hello_SOURCES=hello.c
            hello_LDADD=../lib/libhello.a
            第二行指定頭文件的位置,-I 是idirafter 的縮寫。../include 指定頭文件的位置,..是上
            一級目錄,也就是這里的example 目錄。
            第三行指定生成可執行文件名hello,在這里可執行文件生成在src 下,建議將可執行文
            件生成到一個特定的文件夾下,讓它和源代碼分開,如/root/test 目錄下。寫法為:
            bin_PROGRAMS=/root/test/hello,后面的第四、五行也相對應地變為:
            _root_test_hello_SOURCES=hello.c
            _root_test_hello_LDADD=../lib/libhello.a
            第四行指定生成可執行文件hello 的源代碼文件,如果hello.c 在其他目錄下,需要加上
            完整的路徑。
            第五行指定需要使用靜態庫的位置。
            5. 生成靜態庫文件lib/libhello.a。
            執行autoscan 生成configure.scan 文件,將它重命名為configure.in 并修改其內容。
            #configure.in
            # Process this file with autoconf to produce a configure script.
            AC_PREREQ(2.59)
            AC_INIT(libhello.a,1.1,[])
            AM_INIT_AUTOMAKE
            # Checks for programs.
            AC_PROG_CC
            # Checks for libraries.
            AC_PROG_RANLIB//需要加入的內容,因為使用了靜態庫
            # Checks for header files.
            # Checks for typedefs, structures, and compiler characteristics.
            # Checks for library functions.
            AC_OUTPUT([Makefile])
            AC_INIT(FILE)
            該宏用來檢查源代碼所在路徑,autoscan 會自動產生,一般無須修改它。
            AM_INIT_AUTOMAKE(PACKAGE,VERSION)
            這個是使用 Automake 所必備的宏,PACKAGE 是所要產生軟件的名稱,VERSION 是版
            本編號。也可以把包和版本號等信息放在AC_INIT(FILE) 宏里。
            AC_PROG_CC
            檢查系統可用的C 編譯器,若源代碼是用C 寫的就需要這個宏。
            AC_OUTPUT(FILE)
            設置 configure 所要產生的文件,若是Makefile ,configure 便會把它檢查出來的結果
            填充到Makefile.in 文件后產生合適的 Makefile。 后面的FILE 是一個Makefile 的輸出列表,
            你可以選著將要輸出的Makefile 的位置和個數。建議只在src 中輸出Makefile。
            在lib 目錄下依次執行 aclocal 、autoconf、automake --add-missing、./configure、make,
            此時在該目錄下就可以看到生成的靜態庫文件libhello.a
            6. 在src 目錄下,執行autoscan 生成configure.scan 文件,將它重命名為configure.in 并修
            改其內容。
            #configure.in
            # Process this file with autoconf to produce a configure script.
            AC_PREREQ(2.59)
            AC_INIT(hello,1.1,[])
            AM_INIT_AUTOMAKE
            AC_CONFIG_SRCDIR([hello.c])
            # Checks for programs.
            AC_PROG_CC
            # Checks for libraries.
            # Checks for header files.
            # Checks for typedefs, structures, and compiler characteristics.
            # Checks for library functions.
            AC_OUTPUT([Makefile])
            7. 在src 目錄下依次執行 aclocal 、autoconf、automake --add-missing、./configure、make,
            生成可執行文件hello
            8. 執行make install 進行安裝,最后輸入hello 來運行程序,查看效果:
            Hello static library!
            執行成功!
            使用gcc 創建和使用靜態庫
            1. 編寫mylib.h 文件
            #ifndef _mylib_h_
            #define _mylib_h_
            void welcome();
            void outstring(const char * str);
            #endif
            2. 編寫mylib.c 文件,用來生成靜態庫。
            #include <stdio.h>
            void welcome()
            {
            printf(“welcome to libmylib\n”);
            }
            void outstring(const char * str)
            {
            if(str!=NULL)
            printf(“%s”,str);
            }
            3. 編譯源文件,產生目標代碼
            gcc –o mylib.o –c mylib.c
            4. 將上面產生的目標文件加入到靜態庫中,并把靜態庫拷貝到系統默認的路徑
            ar rcs libmylib.a mylib.o
            cp libmylib.a /usr/lib/
            5. 編寫測試程序來使用剛才創建的靜態庫 libmylib.a
            #include “mylib.h”
            #include <stdio.h>
            Int main()
            {
            printf(“create and use library:\n”);
            welcome();
            outstring(“It’s a successful\n”);
            }
            6. 編譯使用庫函數的程序
            gcc –o test test.c -lmylib
            運行./test 查看結果。


            使用Automake 創建和使用動態庫


            動態庫與靜態庫的差別在于:動態庫是在程序執行的時候加載到內存,供調用函數使用。
            1. 目錄結構如下:
            example
            |——src 目錄(存放源代碼文件)
            |——hello.c
            |——lib 目錄(存放用來生成庫的文件)
            |——test.c 用來生成動態庫libhello.la
            |——include 目錄(存放程序中使用的頭文件)
            |——hello.h
            2. 編寫各個目錄下的源文件如下:
            hello.h 文件
            extern void print(char *);
            test.c 文件
            #include<stdio.h>
            void print(char *msg)
            {
            print(“%s\n”, msg);
            }
            hello.c 文件
            #include “hello.h”
            int main()
            {
            print(“Hello static library!”);//這里用到的是動態庫中的函數
            return 0;
            }
            3. 在lib 目錄下編譯需要生成動態庫的文件,生成動態庫,并安裝到系統的標準庫中,供
            程序調用。具體步驟如下:
            (1) 編寫Makefile.am 文件
            AUTOMAKE_OPTIONS=foreign
            lib_LTLIBRARIES=libhello.la
            libhello_la_SOURCES=test.c
            這里lib_LTLIBRARIES 的意思是生成的動態庫,然后指定動態庫依賴的源文件
            test.c ,若有多個源文件用空格隔開。
            (2) 在lib 目錄下,用命令autoscan 產生configure.scan 文件,并改名為configure.in。 這
            里需加上宏AC_PROG_LIBTOOL,表示利用libtool 來自動生成動態庫
            #configure.in
            # Process this file with autoconf to produce a configure script.
            AC_PREREQ(2.59)
            AC_INIT(hello,1.0, [miaoquan@nou.com.cn])
            AM_INIT_AUTOMAKE
            AC_CONFIG_SRCDIR([test.c])
            #AC_CONFIG_HEADER([config.h])
            # Checks for programs.
            AC_PROG_CC
            # Checks for header files.
            # Checks for typedefs, structures, and compiler characteristics.
            # Checks for library functions.
            AC_PROG_LIBTOOL
            AC_CONFIG_FILES([Makefile])
            AC_OUTPUT
            (3) 執行命令aclocal、libtoolize -f -c 、autoconf、automake --add-missing、./configure、
            make、make install 將動態庫安裝到系統的標準庫中,以供調用(一般為/usr/local/lib)。
            注:libtoolize 提供了一種標準的方式來將libtool 支持加入一個軟件包,而GNU libtool 是
            一個通用庫支持腳本,將使用動態庫的復雜性隱藏在統一、可移植的接口中。
            4. 生成src 目錄下的hello 可執行文件
            (1) 編寫src/Makefile.am 文件
            AUTOMAKE_OPTIONS=foreign
            INCLUDES= -I../include
            bin_PROGRAMS=hello
            hello_SOURCES=hello.c
            hello_LDADD=-lhello
            -ldir 指定編譯時搜索庫的路徑。與靜態庫不同的是,創建動態庫時不用指定庫路
            徑,編譯器自動在標準庫中查找libhello.so 文件。
            (2) 執行autoscan 生成configure.scan 文件,將它重命名為configure.in 并修改其內容。
            # configure.in
            # Process this file with autoconf to produce a configure script.
            AC_PREREQ(2.59)
            AC_INIT(hello,1.0, [miaoquan@nou.com.cn])
            AM_INIT_AUTOMAKE
            AC_CONFIG_SRCDIR([hello.c])
            #AC_CONFIG_HEADER([config.h])
            # Checks for programs.
            AC_PROG_CC
            # Checks for header files.
            # Checks for typedefs, structures, and compiler characteristics.
            # Checks for library functions.
            AC_CONFIG_FILES([Makefile])
            AC_OUTPUT
            (3) 在src 目錄下編譯并生成目標文件,執行命令aclocal、libtoolize -f -c 、autoconf、
            automake --add-missing、./configure、make,此時你一定會覺得,成功近在咫尺了。再
            執行目標文件./hello,結果卻在你的意料之外:
            ./hello: error while loading shared libraries: libhello.so.0 : cannot open shared object file:
            No such file or directory
            在執行目標文件的時候,Shell 找不到共享庫的位置,需要我們手工載入庫路徑。
            5. shell 搜索動態庫路徑位置的兩種方法
            (1) 使用命令導入動態庫的路徑,命令如下:
            export LD_LIBRARY_PATH=dir (如/usr/local/lib)
            (2) 修改/etc/ld.so.conf 文件,加入搜索路徑,修改后用ldconfig 命令載入修改。
            將自己可能存放庫文件的路徑都加入到/etc/ld.so.conf 中是明智的選擇 ^_^。添加
            方法也極其簡單,將庫文件的絕對路徑直接寫進去就OK 了,一行一個。例如:
            /usr/local/lib
            /usr/lib
            /lib
            需要注意的是:這種搜索路徑的設置方式對于程序連接時的庫(包括共享庫和靜態
            庫)的定位已經足夠了,但是對于使用了共享庫的程序的執行還是不夠的。這是 因為
            為了加快程序執行時對共享庫的定位速度,避免使用搜索路徑查找共享庫的低效率,所
            以是直接讀取庫列表文件 /etc/ld.so.cache 從中進行搜索的。/etc/ld.so.cache 是一個非
            文本的數據文件,不能直接編輯,它是根據 /etc/ld.so.conf 中設置的搜索路徑由
            /sbin/ldconfig 命令將這些搜索路徑下的共享庫文件集中在一起而生成的(ldconfig 命令
            要以 root 權限執行)。因此,為了保證程序執行時對庫的定位,在 /etc/ld.so.conf 中
            進行了庫搜索路徑的設置之后,還必須要運行 /sbin/ldconfig 命令更新 /etc/ld.so.cache
            文件之后才可以。ldconfig ,簡單的說,它的作用就是將/etc/ld.so.conf 列出的路徑下的庫
            文件 緩存到/etc/ld.so.cache 以供使用。因此當安裝完一些庫文件,(例如剛安裝好glib),
            或者修改ld.so.conf 增加新的庫路徑后,需要運行一下/sbin/ldconfig 使所有的庫文件都
            被緩存到ld.so.cache 中,如果沒做,即使庫文件明明就在/usr/lib 下的,也是不會被使
            用的,結果編譯過程中報錯,缺少xxx 庫,去查看發現明明就在那放著,搞的想大罵
            computer 蠢豬一個^_^。極力推薦使用這種方法!
            利用gcc 創建和使用動態庫
            1. 用下面的命令將mylib.c 程序創建成一個動態庫:
            gcc –fPIC –o mylib.o –c mylib.c
            gcc –shared –o libtt.so mylib.o
            -fPIC 作用于編譯階段,告訴編譯器產生與位置無關代碼(Position-Independent Code),
            則產生的代碼中,沒有絕對地址,全部使用相對地址,故而代碼可以被加載器加載到內存的
            任意位置,都可以正確的執行。這正是共享庫所要求的,共享庫被加載時,在內存的位置不
            是固定的。
            -shared 作用于鏈接階段,實際傳遞給鏈接器ld,讓其添加作為共享庫所需要的額外描
            述信息,去除共享庫所不需的信息。
            也可以直接使用下面一條命令:
            gcc –fPIC –shared –o libtt.so mylib.c
            2. 將動態庫拷貝到linux 的標準庫中,usr/local/lib 或者/usr/lib 或者/lib:
            cp libttt.so /usr/local/lib
            3. 編譯src 目錄下的源程序時,指定動態庫文件的目錄,調用動態庫中的函數
            gcc –o test test.c /usr/lib/libttt.so
            4. 設置shell 動態庫搜索路徑,運行生成的可執行文件。
            ---------------------------------------------------------------------------

            AUTOMAKE_OPTIONS=foreign

            INCLUDES=-I$(top_srcdir)/libpr/include -I$(top_srcdir)/vt/include

            noinst_PROGRAMS =libvt.so
            libvt_so_SOURCES=vty/vtcmd.c vty/vtdrv.c vty/vty.c vty/evtd.c telnet/telcmd.c telnet/teldrv.c telnet/telnegot.c telnet/telsvr.c telnet/telsess.c vty/defcmd.c vty/vtobj.c
            libvt_so_LDFLAGS = -fPIC -shared
            libvt_so_LDADD=$(top_srcdir)/libpr/libpr.a

            bin_PROGRAMS = test_telnetd
            test_telnetd_SOURCES = test/test_telnetd.c 
            test_telnetd_LDADD=$(top_srcdir)/libpr/libpr.a libvt.so

            上面這段代碼是從我的一個工程的Makefile.am中摘抄過來的,使用不少技巧。

            1、動態庫實際上也是ELF格式,所以我們使用PROGRAMS宏,automake將他按執行文件規格設置環境,如果使用noinstall_LIBRARIES那么就變成*.a靜態庫了。

            2、因為是動態庫,所以我們有必要加入-fPIC -shared

            3、test_telnetd需要調用libvt.so,所以他們之間存在依賴關系。我們將libvt.so直接添加在LDADD 中,automake會自動他們建立依賴關系,在Makefile中可以到test_telnetd_DEPENDENCIES,里面包含 libvt.so。

            這里有幾個要緊需要注意:

            1、$(bin_PROGRAMS)編譯順序在$(noinstall_PROGRAMS)之前,因為test_telnetd需要libvt.so,所以會找不到libvt.so。

            2、在LDADD中如果使用-lvt不會建立依賴關系,如果使用$(top_srcdir)/vt/libvt.so也不行。

            3、實際上如果不行的話,我們可以直接使用test_telnetd_DEPENDENCIES來指定libvt.so

            4、如果需要make install 安裝libvt.so到lib,那么使用XXX_PROGRAMS=...的方法,然后指定XXXbin=...就行了。類似于bin_PROGRAMS。

            這里使用到的技巧都很有用,值得記錄下來。

            posted on 2011-06-01 13:22 艾斯維亞 閱讀(5628) 評論(0)  編輯 收藏 引用

            欧美色综合久久久久久| 亚洲嫩草影院久久精品| 久久99久国产麻精品66| 东方aⅴ免费观看久久av| 国产精品久久波多野结衣| 精品久久人人做人人爽综合 | 久久国产色av免费看| 久久久久久久久无码精品亚洲日韩 | 日韩欧美亚洲综合久久| 国产精品女同久久久久电影院| 国产精品欧美久久久久无广告| 伊人久久无码精品中文字幕| 久久久久亚洲AV无码麻豆| 狠狠久久综合伊人不卡| 久久精品国产亚洲αv忘忧草| 久久精品一区二区三区不卡| 亚洲国产成人久久一区久久| 久久婷婷综合中文字幕| 色欲av伊人久久大香线蕉影院| 91亚洲国产成人久久精品| 少妇高潮惨叫久久久久久| 久久伊人亚洲AV无码网站| 国产精品久久永久免费| 亚洲精品美女久久久久99| 亚洲精品综合久久| 久久青青草原精品国产不卡| 亚洲嫩草影院久久精品| AV无码久久久久不卡蜜桃| 一本久久知道综合久久| 久久天天躁狠狠躁夜夜av浪潮| 狠狠色丁香久久婷婷综合五月| 久久精品国产亚洲AV香蕉| 亚洲国产小视频精品久久久三级 | 狠狠色噜噜狠狠狠狠狠色综合久久 | 久久精品九九亚洲精品| 日韩精品久久久肉伦网站| 亚洲AV日韩精品久久久久久久| 精品久久久无码21p发布| 区亚洲欧美一级久久精品亚洲精品成人网久久久久 | 51久久夜色精品国产| 国产韩国精品一区二区三区久久|