草庐IT

AAR依赖和module源码动态切换

ingxin 2023-03-28 原文

痛点

壳工程通过implementation 'com.alibaba:fastjson:1.2.76'的形式引入aar文件,而aar使用一个单独的业务工程开发,这种形式开发模式常见于组件化的工程中。这样做可以隔离代码,深度解耦,业务复用,节省编译时间。然而有时候我们需要在壳工程中进行aar联调,这时候我们就需要把aar工程的源码引入到壳工程中,在壳工程中做法如下:

  1. settings.gradle文件中添加如下配置

    include ":moduleName"
    project(":moduleName").projectDir = file("源码路径")
    
  2. 在app模块通过implementation project(":moduleName")方式引入

  3. 调试好后移除上面配置,发布版本。

这样做虽然能达到目的,但不够优雅,存在忘记恢复导致CI不能正确打包的可能性。下面我们进行优雅改造,文中所用的是AGP7.0+,AGP7.0改动挺大,7.0以下的自己修改即可。

步骤2可以使用gradle提供的替换apisubstitute module(dependenceName) with project(":moduleName"),该api可以将原始依赖dependenceName(例如com.alibaba:fastjson,不包含版本号),替换成本地moduleName模块。
可以监听app模块下的gradle配置

configurations.all { config ->
    config.resolutionStrategy.dependencySubstitution {
      //使用substitute module api
    }    
 }

这样就不用反复修改依赖了,但这样还不足够方便

优雅永不过时

在工程根目录下新增debug_aar.gradle脚本,脚本内容如下。(AS提示import com.alibaba.fastjson.JSONArray错误,实际上是成功导入的,知道怎么消除的留言)

//https://docs.gradle.org/current/userguide/tutorial_using_tasks.html
buildscript {
    //依赖仓库源
    repositories {
        maven { url 'https://maven.aliyun.com/nexus/content/repositories/google' }
        maven { url 'https://maven.aliyun.com/nexus/content/groups/public' }
        mavenCentral()

    }
    dependencies {
        //为当前脚本添加解析gson的依赖
        classpath "com.alibaba:fastjson:1.2.76"
    }
}

//AS会提示找不到JSONArray,JSONObject直接忽略
import com.alibaba.fastjson.JSONArray
import com.alibaba.fastjson.JSONObject

List<ModuleSource> list = loadDebugConfig()

for (ModuleSource module : list) {
    if (module.debug) {
        include ":${module.moduleName}"
        //gradle8弃用了/xx/xx相对路径的形式,所以用使用$绝对路径
        project(":${module.moduleName}").projectDir = file("${module.sourceDir}")
        println("debug外部模块[:${module.moduleName}],源码路径 ${module.sourceDir}")
    }
}

if (list.size() > 0) {
    gradle.addProjectEvaluationListener(new ProjectEvaluationListener() {
        @Override
        void beforeEvaluate(Project projectObj) {
        }

        @Override
        void afterEvaluate(Project projectObj, ProjectState state) {
//            boolean isAppModule = projectObj.plugins.hasPlugin('com.android.application')
//            if (!isAppModule) {
//                //只处理application模块
//                return
//            }
            projectObj.configurations.all { config ->
                config.resolutionStrategy.dependencySubstitution {
                    for (ModuleSource ms : list) {
                        if (ms.debug) {
                            substitute module(ms.dependenceName) with project(":${ms.moduleName}")
                        }
                    }
                }
            }
        }
    })
}

def loadDebugConfig() {
    List<ModuleSource> list = new ArrayList<>()
    String json = null
    try {
        json = file("debug_aar_config.json").getText()
    } catch (ignored) {
        println("根目录不存在debug_aar_config.json文件。(如果不需要debug aar源码忽略该信息)")
    }
    if (json == null) {
        return list
    }

    //解析debug_source_config.json中的字段
    JSONArray jsonArray = (JSONArray) JSONObject.parse(json)
    for (int i = 0; i < jsonArray.size(); i++) {
        JSONObject ob = (JSONObject) jsonArray.get(i)
        boolean isDebug = ob.getBoolean("debug")
        String moduleName = ob.getString("module_name")
        String sourceDir = ob.getString("source_dir")
        String dependenceName = ob.getString("dependence_name")
        if (moduleName == null || sourceDir == null || dependenceName == null) {
            println("数据[${moduleName},${sourceDir},${dependenceName}]异常,该配置被忽略!!")
            continue
        }
        list.add(new ModuleSource(isDebug, moduleName, sourceDir, dependenceName))
    }
    return list
}

class ModuleSource {

    /**是否调试aar*/
    boolean debug = false

    /**引入module名字*/
    String moduleName = null

    /**
     * aar依赖,去掉版本号,例如引入aar依赖com.google.code.gson:gson:2.8.5
     * 则dependenceName为com.google.code.gson:gson
     */
    String dependenceName = null

    /**绝对路径*/
    String sourceDir = null

    ModuleSource(boolean debug, String moduleName, String sourceDir, String dependenceName) {
        this.debug = debug
        this.moduleName = moduleName
        this.dependenceName = dependenceName
        this.sourceDir = sourceDir
    }
}

使用

settings.gradle中引入上面的脚本,如下:

rootProject.name = "Test"
include ':app'

//引入脚本
try {
    apply from: 'debug_aar.gradle'
} catch (ignored) {
}

让这个脚本工作起来还需要一个配置debug_aar_config.json文件,这个文件应该被加入到git忽略文件,如果不存在这个json文件,我们的脚本是不工作的,这样就不影响CI的正式发版。
在根目录下创建debug_aar_config.json文件,文件内容如下

[
  {
    "debug": true,
    "module_name": "testModdule",
    "source_dir": "D:\xxx\test",
    "dependence_name": "com.alibaba:fastjson"
  }
]
  • debug:是否调试aar源码
  • module_name:调试源码时模块的名字
  • source_dir:源码路径,以根目录为起点的路径写法,所以一般需要用..返回到上级
  • dependence_name:aar的远程依赖
    json是个数组,可以增加或者删除,每次修改完json文件,需要sync工程通知修改。
打开调试
关闭调试

可以看到只需修改json即可。

注意事项

  1. org.gradle.configureondemand=true的配置,会导致在调试aar源码时,找不到对应的module。所以需要关闭这个配置

有关AAR依赖和module源码动态切换的更多相关文章

  1. ruby - Facter::Util::Uptime:Module 的未定义方法 get_uptime (NoMethodError) - 2

    我正在尝试设置一个puppet节点,但ruby​​gems似乎不正常。如果我通过它自己的二进制文件(/usr/lib/ruby/gems/1.8/gems/facter-1.5.8/bin/facter)在cli上运行facter,它工作正常,但如果我通过由ruby​​gems(/usr/bin/facter)安装的二进制文件,它抛出:/usr/lib/ruby/1.8/facter/uptime.rb:11:undefinedmethod`get_uptime'forFacter::Util::Uptime:Module(NoMethodError)from/usr/lib/ruby

  2. ruby-on-rails - Ruby on Rails with Haml - 如何从 erb 切换 - 2

    我正在从erb文件切换到HAML。我将hamlgem添加到我的系统中。我创建了app/views/layouts/application.html.haml文件。我应该只删除application.html.erb文件吗?此外,仍然有/public/index.html文件被呈现为默认页面。我想创建自己的默认index.html.haml页面。我应该把它放在哪里以及如何使系统呈现该文件而不是默认索引文件?谢谢! 最佳答案 是的,您可以删除任何已转换为HAML的View的ERB版本。至于你的另一个问题,删除public/index/h

  3. ruby - 为什么人们使用 `Module.send(:prepend, …)` ? - 2

    我正在学习如何在我的Ruby代码中使用Module.prepend而不是alias_method_chain,我注意到有些人使用send调用它(example):ActionView::TemplateRenderer.send(:prepend,ActionViewTemplateRendererWithCurrentTemplate)而其他人直接调用它(example):ActionView::TemplateRenderer.prepend(ActionViewTemplateRendererWithCurrentTemplate)而且,虽然我还没有看到任何人使用这种风格,但我从

  4. UE4 源码阅读:从引擎启动到Receive Begin Play - 2

    一、引擎主循环UE版本:4.27一、引擎主循环的位置:Launch.cpp:GuardedMain函数二、、GuardedMain函数执行逻辑:1、EnginePreInit:加载大多数模块int32ErrorLevel=EnginePreInit(CmdLine);PreInit模块加载顺序:模块加载过程:(1)注册模块中定义的UObject,同时为每个类构造一个类默认对象(CDO,记录类的默认状态,作为模板用于子类实例创建)(2)调用模块的StartUpModule方法2、FEngineLoop::Init()1、检查Engine的配置文件找出使用了哪一个GameEngine类(UGame

  5. ruby-on-rails - 测试我的 Ruby gem:Shoulda::Matchers:Module (NoMethodError) 的未定义方法 `configure' - 2

    我正在开发我的第一个Rubygem,并捆绑了cucumber、rspec和shoulda-matches进行测试。当我运行rspec时,出现以下错误:/app/my_gem/spec/spec_helper.rb:6:in`':undefinedmethod`configure'forShoulda::Matchers:Module(NoMethodError)这是我的gem规范:#my_gem.gemspec...Gem::Specification.newdo|spec|......spec.add_development_dependency"activemodel"spec.a

  6. ruby - 在 Ruby 中动态创建数组 - 2

    有没有办法在Ruby中动态创建数组?例如,假设我想遍历用户输入的书籍数组:books=gets.chomp用户输入:"TheGreatGatsby,CrimeandPunishment,Dracula,Fahrenheit451,PrideandPrejudice,SenseandSensibility,Slaughterhouse-Five,TheAdventuresofHuckleberryFinn"我把它变成一个数组:books_array=books.split(",")现在,对于用户输入的每一本书,我想用Ruby创建一个数组。伪代码来做到这一点:x=0books_array.

  7. ruby - 是否可以将 IRB 提示配置为动态更改? - 2

    我想在IRB中浏览文件系统并让提示更改以反射(reflect)当前工作目录,但我不知道如何在每个命令后进行提示更新。最终,我想在日常工作中更多地使用IRB,让bash溜走。我在我的.irbrc中试过这个:require'fileutils'includeFileUtilsIRB.conf[:PROMPT][:CUSTOM]={:PROMPT_N=>"\e[1m:\e[m",:PROMPT_I=>"\e[1m#{pwd}>\e[m",:PROMPT_S=>"FOO",:PROMPT_C=>"\e[1m#{pwd}>\e[m",:RETURN=>""}IRB.conf[:PROMPT_MO

  8. ruby-on-rails - carrierwave:在序列化动态属性上安装 uploader - 2

    首先,我使用的是rails3.1.3和来自master的carrierwavegithub仓库的分支。我使用after_init钩子(Hook)来确定基于属性的字段页面模型实例并为这些字段定义属性访问器将值存储在序列化哈希中(希望它清楚我是什么谈论)。这是我正在做的事情的精简版:classPage省略mount_uploader命令让我可以访问我想要的属性。但是当我安装uploader时出现错误消息说“nil类的未定义新方法”我在源代码中读到有方法read_uploader和扩展模块中的write_uploader。我如何必须覆盖这些来制作mount_uploader命令使用我的“虚拟

  9. ruby - 在 Ruby 中动态生成多维数组 - 2

    我正在尝试动态构建一个多维数组。我想要的基本上是这样的(为简单起见写出来):b=0test=[[]]test[b]这给了我错误:NoMethodError:undefinedmethod`test=[[],[],[]]而且它工作正常,但在我的实际使用中,我不会事先知道需要多少个数组。有一个更好的方法吗?谢谢 最佳答案 不需要像您正在使用的索引变量。只需将每个数组附加到您的test数组:irb>test=[]=>[]irb>test[["a","b","c"]]irb>test[["a","b","c"],["d","e","f"]]

  10. ruby-on-rails - 使用 gmaps4rails 动态加载谷歌地图标记 - 2

    如何只加载map边界内的标记gmaps4rails?当然,在平移和/或缩放后加载新的。与此直接相关的是,如何获取map的当前边界和缩放级别? 最佳答案 我是这样做的,我只在用户完成平移或缩放后替换标记,如果您需要不同的行为,请使用不同的事件监听器:在你看来(index.html.erb):{"zoom"=>15,"auto_adjust"=>false,"detect_location"=>true,"center_on_user"=>true}},false,true)%>在View的底部添加:functiongmaps4rail

随机推荐