• makefile学习经验(二)编译生成静态库文件


    Lunix下编译静态库文件:

    .o后缀文件:编译生成的中间代码文件;

    .a后缀文件:静态库文件,编译的时候会合到可执行程序中,文件比较大;

    .so后缀文件:动态库文件,只是一个指向,不会合到可执行程序中,当要调用函数库的时候才使用;

    用一个简单的例子阐述下在lunix环境下生成静态库文件的过程:

    一.   源码文件:

    so_test.h:

    void test_a();

    void test_b();

    void test_c();

    test_a.c:

    #include "so_test.h"

    void test_a()

    {

        printf("this is in test_a...\n");

    }

    test_b.c:

    #include "so_test.h"

    void test_a()

    {

        printf("this is in test_b...\n");

    }

    test_c.c:

    #include "so_test.h"

    void test_a()

    {

        printf("this is in test_c...\n");

    }

    test.c:

    #include "so_test.h"

    int main()

    {

    test_a();

    test_b();

    test_c();

    return 0;

    }

    二.   makefile 文件:

    makefile:

    #*****************************************************************************

    # Copyright        :  

    #

    # Author           :   kimi

    # Date             :   2012-08-17

    # Version          :   Demo2 静态链接库

    # Description  :   Demo

    #

    #****************************************************************************/

    SHELL = /bin/sh

    LIB_DIR = /cbs/lhbb/hcj/dy/lib/

    BIN_DIR = /cbs/lhbb/hcj/dy/bin/

    OBJECT_DIR = /cbs/lhbb/dy/test/obj/

    APP_DIR = /cbs/lhbb/hcj/dy/testapp/

    $(shell mkdir -p ${LIB_DIR})

    $(shell mkdir -p ${BIN_DIR})

    $(shell mkdir -p ${OBJECT_DIR})

    RM = rm -fr

    #****************************************************************************

    CC = gcc

    AR = ar rc

    SRC_OBJECT = $(APP_DIR)test_a.c $(APP_DIR)test_b.c $(APP_DIR)test_c.c

    H_OBJECT = $(APP_DIR)so_test.h

    OBJECT = test_a.o test_b.o test_c.o

    DY_SRC_OBJECT = $(APP_DIR)test.c

    DY_OBJECT=test.o

    LIB_OBJECT = libtest.a

    BIN_OBJECT = test

    #****************************************************************************

    .PHONY:all

    all:$(LIB_OBJECT) $(BIN_OBJECT)

    $(LIB_OBJECT):$(OBJECT)

            $(AR) $(LIB_OBJECT) $(OBJECT)

            mv $(LIB_OBJECT) $(LIB_DIR)

    $(OBJECT):$(SRC_OBJECT) $(H_OBJECT)

            $(CC) -c $(SRC_OBJECT)

    $(BIN_OBJECT):$(DY_OBJECT)

            $(CC) -o $(BIN_OBJECT) $(OBJECT_DIR)$(DY_OBJECT) $(LIB_DIR)$(LIB_OBJECT)

            mv $(BIN_OBJECT) $(BIN_DIR)

    $(DY_OBJECT):$(DY_SRC_OBJECT)

            $(CC) -c $(DY_SRC_OBJECT) -o $(DY_OBJECT)

            mv $(OBJECT) $(DY_OBJECT) $(OBJECT_DIR)

    clean:

            $(RM) $(LIB_DIR) $(BIN_DIR) $(OBJECT_DIR)

    三.makefile文件的简单阐述:

    1. 路径变量赋值:

    LIB_DIR = /cbs/lhbb/hcj/dy/lib/          -----存放生成的静态库文件,.a文件;

    BIN_DIR = /cbs/lhbb/hcj/dy/bin/         -----存放生成的可执行文件;

    OBJECT_DIR = /cbs/lhbb/dy/test/obj/     -----存放生成的中间代码文件,.o文件;

    APP_DIR = /cbs/lhbb/hcj/dy/testapp/     -----存放源文件;

    2.调用shell脚本,创建目录:

    $(shell mkdir -p ${LIB_DIR})

    $(shell mkdir -p ${BIN_DIR})

    $(shell mkdir -p ${OBJECT_DIR})

    3.编译变量赋值:

    RM = rm –fr

    CC = gcc

    AR = ar rc       -----生成静态库文件命令

    4.文件变量赋值:

    SRC_OBJECT = $(APP_DIR)test_a.c $(APP_DIR)test_b.c $(APP_DIR)test_c.c

    ----源文件变量;

    H_OBJECT = $(APP_DIR)so_test.h

    ----头文件变量;

    OBJECT = test_a.o test_b.o test_c.o

    ----中间代码文件变量;

    DY_SRC_OBJECT = $(APP_DIR)test.c

    ----生成test可执行文件所依赖的源文件变量;

    DY_OBJECT=test.o

    ----编译test.c生成的中间代码文件变量;

    LIB_OBJECT = libtest.a

    ----静态库文件变量;

    BIN_OBJECT = test

    ----生成的可执行文件变量;

    这个makefile中基本都是使用的变量,这样编译不同的工程的时候,就只要修改变量的值,编译的整体框架就不需要修改,当然,这个框架只是本人刚开始研究makefile写的,不够好,望大神吐槽,谢谢!

    5.编译主体阐述:

    .PHONY:all

    ----固定用.PHONY定义一个伪目标,伪目标不是文件,只是一个标签,没有依赖文件;这边定义了一个伪目标all;

    all:$(LIB_OBJECT) $(BIN_OBJECT)

    ----目标all包含了两个需要生成的目标文件,一个是需要生成的静态库文件libtest.a,一个是最终生成的可执行文件test;但是为什么这边需要用一个all来包含两个目标文件呢?原因是通常情况下,系统只会去编译生成makefile文件开头最早一个目标文件,下面的目标文件是不会去执行编译的,所以我在这边将两个目标文件写在了一个伪目标all中,这样就可以实现生成两个目标文件;

    生成静态库文件的过程:

    $(LIB_OBJECT):$(OBJECT)      -----目标文件:依赖文件    libtest.a:test_a.o test_b.o test_c.o

            $(AR) $(LIB_OBJECT) $(OBJECT)    ----编译生成静态库文件命令,特点就是需要使用

                            ar rc命令去编译;

            mv $(LIB_OBJECT) $(LIB_DIR)      ----将生成的文件移动到对应的文件夹;

    $(OBJECT):$(SRC_OBJECT) $(H_OBJECT)

            $(CC) -c $(SRC_OBJECT)

    生成可执行文件test的过程:

    $(BIN_OBJECT):$(DY_OBJECT)

            $(CC) -o $(BIN_OBJECT) $(OBJECT_DIR)$(DY_OBJECT) $(LIB_DIR)$(LIB_OBJECT)

            -----将静态库文件包含在命令中;

            mv $(BIN_OBJECT) $(BIN_DIR)

    $(DY_OBJECT):$(DY_SRC_OBJECT)

            $(CC) -c $(DY_SRC_OBJECT) -o $(DY_OBJECT)

            mv $(OBJECT) $(DY_OBJECT) $(OBJECT_DIR)

    执行make clean删除生成的文件:

    clean:

            $(RM) $(LIB_DIR) $(BIN_DIR) $(OBJECT_DIR)

    四.注意点:

    1.命令行必须以tab键打头,否则会报错;

    2.生成静态库文件用命令:ar rc;

    3.库文件以lib打头,.a为后缀;

    五.执行结果:

    bin文件夹中存放可执行文件test,运行./test结果如下:

    this is in test_a...

    this is in test_b...

    this is in test_c...

    lib文件夹下存放生成的静态库文件libtest.a;

  • 相关阅读:
    Python数据可视化的四种简易方法
    阿里如何将“高峰前扩容、高峰后缩容”的梦想照进现实?
    支付宝客户端架构解析:Android 容器化框架初探
    阿里工业互联网平台“思考”:一场从0到1的蜕变
    关于Flutter初始化流程,我必须告诉你的是...
    类在什么时候加载和初始化
    instanceof 与isAssignableFrom
    Java注释@interface的用法【转】
    网站过滤器Filter
    正则表达式
  • 原文地址:https://www.cnblogs.com/huochangjun/p/2670150.html
Copyright © 2020-2023  润新知