zoukankan      html  css  js  c++  java
  • 第23课

    1.问题背景

       一般而言,在一个大型项目中不同工程师负责不同模块的开发;那么问题就来了,我们怎么知道自己编写的这部分代码有没有问题,怎样进行编译?在编译环境中如何支持模块的独立编译

    • 在大型项目中的代码文件成千上万,完整编译的时间较长

    • 编写模块代码时可通过编译检查语法错误

    • 为了提高开发效率需要支持指定模块的独立编译

        

    2.解决方案

    1、将模块名(module)作为目标名(伪目标)建立规则;

    2、目标(module)对应的依赖为 build build/module

    3、规则中的命令进入对应的模块文件夹进行编译

    4、编译结果存放于 build 文件夹下。

    关键技术点:

    如何获取 make 命令行指定编译的模块名

    • 通过预定义变量:$(MAKECMDGOALS)命令行指定的目标名(make 的命令行参数)

          图片.png

    下来我们来看看具体的 makefile 是怎样写的,将上节博客中的 pro-rule.mk 改成下面这样:

    模块的独立编译:

    .PHONY : all compile link clean rebuild $(MODULES)
    
    DIR_PROJECT := $(realpath .)
    DIR_BUILD_SUB := $(addprefix $(DIR_BUILD)/, $(MODULES))
    MODULE_LIB := $(addsuffix .a, $(MODULES))
    MODULE_LIB := $(addprefix $(DIR_BUILD)/, $(MODULE_LIB))
    
    APP := $(addprefix $(DIR_BUILD)/, $(APP))
    
    all : compile $(APP)
        @echo "Success! Target ==> $(APP)"
    
    compile : $(DIR_BUILD) $(DIR_BUILD_SUB)
        @echo "Begin to compile ..."
        @set -e; 
        for dir in $(MODULES); 
        do 
            cd $$dir && 
            $(MAKE) all 
                DEBUG:=$(DEBUG) 
                DIR_BUILD:=$(addprefix $(DIR_PROJECT)/, $(DIR_BUILD)) 
                DIR_COMMON_INC:=$(addprefix $(DIR_PROJECT)/, $(DIR_COMMON_INC)) 
                CMD_CFG:=$(addprefix $(DIR_PROJECT)/, $(CMD_CFG)) 
                MOD_CFG:=$(addprefix $(DIR_PROJECT)/, $(MOD_CFG)) 
                MOD_RULE:=$(addprefix $(DIR_PROJECT)/, $(MOD_RULE)) && 
            cd .. ; 
        done
        @echo "Compile Success!"
        
    link $(APP) : $(MODULE_LIB)
        @echo "Begin to link ..."
        $(CC) -o $(APP) -Xlinker "-(" $^ -Xlinker "-)" $(LFLAGS)
        @echo "Link Success!"
        
    $(DIR_BUILD) $(DIR_BUILD_SUB) : 
        $(MKDIR) $@
        
    clean : 
        @echo "Begin to clean ..."
        $(RM) $(DIR_BUILD)
        @echo "Clean Success!"
        
    rebuild : clean all
    
    $(MODULES) : $(DIR_BUILD) $(DIR_BUILD)/$(MAKECMDGOALS)
        @echo "Begin to compile $@"
        @set -e; 
        for dir in $(MODULES); 
        do 
            cd $@ && 
            $(MAKE) all 
                DEBUG:=$(DEBUG) 
                DIR_BUILD:=$(addprefix $(DIR_PROJECT)/, $(DIR_BUILD)) 
                DIR_COMMON_INC:=$(addprefix $(DIR_PROJECT)/, $(DIR_COMMON_INC)) 
                CMD_CFG:=$(addprefix $(DIR_PROJECT)/, $(CMD_CFG)) 
                MOD_CFG:=$(addprefix $(DIR_PROJECT)/, $(MOD_CFG)) 
                MOD_RULE:=$(addprefix $(DIR_PROJECT)/, $(MOD_RULE)) && 
            cd .. ; 
        done
        @echo "Compile Success!"

    运行结果:

      图片.png

     我们看到 common 模块已经正确编译了,而且生成相应的 common.a 文件了。我们再继续编译别的两个模块,再通过链接的命令看看可执行程序 app.out 是否可以生成。

      图片.png

            我们看到可执行程序 app.out 已经正确生成了。那么我们看到刚才的模块编写是直接复制之前的代码,凡是涉及到复制粘贴的代码,我们得看看是否可以封装成类似于函数的形式

    makefile中的代码复用:

    • 当不同规则中的命令大量重复时,可考虑自定义函数

    • makefile 中的自定义函数代码复用的一种方式

            图片.png

    具体思路:

    1、将编译模块的命令作为自定义函数的具体实现;

    2、函数参数模块名,函数调用后编译参数指定的模块;

    3、在不同的规则中调用该函数。如下

        图片.png 

    下面我们看看改变后的 makefile 是怎样的,将前面的 pro-rule.mk 改成下面这样:

    .PHONY : all compile link clean rebuild $(MODULES)
    
    DIR_PROJECT := $(realpath .)
    DIR_BUILD_SUB := $(addprefix $(DIR_BUILD)/, $(MODULES))
    MODULE_LIB := $(addsuffix .a, $(MODULES))
    MODULE_LIB := $(addprefix $(DIR_BUILD)/, $(MODULE_LIB))
    
    APP := $(addprefix $(DIR_BUILD)/, $(APP))
    
    define makemodule
        cd ${1} && 
        $(MAKE) all 
            DEBUG:=$(DEBUG) 
            DIR_BUILD:=$(addprefix $(DIR_PROJECT)/, $(DIR_BUILD)) 
            DIR_COMMON_INC:=$(addprefix $(DIR_PROJECT)/, $(DIR_COMMON_INC)) 
            CMD_CFG:=$(addprefix $(DIR_PROJECT)/, $(CMD_CFG)) 
            MOD_CFG:=$(addprefix $(DIR_PROJECT)/, $(MOD_CFG)) 
            MOD_RULE:=$(addprefix $(DIR_PROJECT)/, $(MOD_RULE)) && 
        cd .. ;
    endef
    
    all : compile $(APP)
        @echo "Success! Target ==> $(APP)"
    
    compile : $(DIR_BUILD) $(DIR_BUILD_SUB)
        @echo "Begin to compile ..."
        @set -e; 
        for dir in $(MODULES); 
        do 
            $(call makemodule, $$dir) 
        done
        @echo "Compile Success!"
        
    link $(APP) : $(MODULE_LIB)
        @echo "Begin to link ..."
        $(CC) -o $(APP) -Xlinker "-(" $^ -Xlinker "-)" $(LFLAGS)
        @echo "Link Success!"
        
    $(DIR_BUILD) $(DIR_BUILD_SUB) : 
        $(MKDIR) $@
        
    clean : 
        @echo "Begin to clean ..."
        $(RM) $(DIR_BUILD)
        @echo "Clean Success!"
        
    rebuild : clean all
    
    $(MODULES) : $(DIR_BUILD) $(DIR_BUILD)/$(MAKECMDGOALS)
        @echo "Begin to compile $@"
        @set -e; 
        $(call makemodule, $@)

    运行结果:

      图片.png

            我们看到 makefile 的代码已经是相当简洁了,通过对单独模块 makefile 的编写。

    3.总结

    1、编写模块代码时可通过模块独立编译快速检查语法错误

    2、自动变量只能在规则的命令中使用不能在依赖中使用

    3、makefile 中的自定义函数代码复用的一种方式

    4、当不同规则中的命令大量重复时可考虑自定义函数

  • 相关阅读:
    Android Studio3.1.2编译时Java Compiler出错:Warning: Failed to parse host proxy3.bj...
    Android RxJava 2 的用法 just 、from、map、subscribe、flatmap、Flowable、Function、Consumer ...【转】
    Android DevArt6:Android中IPC的六种方式
    Android 深入浅出
    Eclipse代码自动补全
    Android 深入浅出
    Android 深入浅出
    Android中查看SQLite中字段数据的两种方式
    Android Studio 打包时 Signature Version 选择V1还是V2 ?
    Android进阶AIDL
  • 原文地址:https://www.cnblogs.com/hoiday/p/10306609.html
Copyright © 2011-2022 走看看