Android 集成Tinker踩坑记录_android tinker-程序员宅基地

技术标签: 补丁管理  android  加固包生成补丁  Tinker  热修复  Android  

背景

热修复早已不是什么新鲜技术了,各个大厂基本都有自己的热修复方案。
关于各个热修方案的对比就不赘述了,网上一搜一大堆。直接看Tinker官方文档 就行。

这里说下为什么选择Tinker吧
先说优点

  1. 开源免费,讲道理,要不是免费肯定用集成起来更简单的Sophix了
  2. 目前仍处于维护状态
  3. 腾讯背书,官方说微信也在用,那我们自然没什么好顾虑的了

说完优点再吐槽下缺点

  1. 文档基本没怎么更新了,项目稍微新一点的集成起来需要踩坑
  2. demo比较老,agp版本很低
  3. 集成使用比较麻烦,代码量相较于常规的其它三方sdk多很多
  4. 处于open状态的issues数量比较多,开源项目遇到奇奇怪怪的问题只能尽量自己解决,肯定不会像商业sdk有技术快速支持的。
  5. 补丁的版本管理和维护需要自己搭建

集成步骤

这里我说下我自己比较推荐的集成步骤

  1. 先把源码下载下来,跑通,能够在demo上实现热修的功能
  2. 新建一个干净的空项目,从0开始集成tinker,同样要实现热修的功能
  3. 最后再往自己的项目中集成

集成步骤官方文档写的也比较清楚了,只是demo的agp版本比较老,一些配置的增加也没有在文档中更新。因此,在高版本的AGP项目中集成起来还是要踩一些坑的。

这里我把我集成tinker时踩的坑记录一下,有些坑也花了点时间去解决,希望能帮到需要的同学。
先说下我的AGP版本,这个可能会影响到Tinker

  1. AGP版本是7.0.4
  2. gradle版本是7.3.3
  3. tinker用的就是最新的版本(目前是:v1.9.14.25.3)

AGP7.x 配置Tinker

我们都知道AGP7开始,之前的classpath配置有所变更,如果你是AGP7以上的版本,需要在settings.gradle添加tinker的配置

pluginManagement {
    
    repositories {
    
        google()
        mavenCentral()
        gradlePluginPortal()
    }

    resolutionStrategy {
    
        /*配置tinker*/
        eachPlugin {
    
            if (requested.id.id == "com.tencent.tinker.patch") {
    
                useModule("com.tencent.tinker:tinker-patch-gradle-plugin:1.9.14.25.3")
            }
        }
    }
}

随后就正常在App模块里的build.gradle添加tinker所需依赖即可
示例:

plugins {
    
    id 'com.android.application'
    id 'org.jetbrains.kotlin.android'
    id 'com.tencent.tinker.patch'//tinker插件
    id 'kotlin-kapt'
}

生成补丁包时提示 Could not resolve all files for configuration ‘:app:sevenZipToolsLocator’.

我是在M1的电脑上生成补丁包,过程中提示找不到7zip。
根据报错的信息 Could not find SevenZip-1.1.10-osx-aarch_64.exe (com.tencent.mm:SevenZip:1.1.10). 中的.exe 也能猜出来是平台相关的问题。
在这里插入图片描述
这里我们可以按照以下步骤做解决这个问题

  1. brew install p7zip 本地安装下7zip
  2. 获取本地安装的7zip路径 which 7za
  3. 更改下配置,把zipArtifact配置注释掉,path更改为本机安装的7za路径
        sevenZip {
    
            /**
             * optional,default '7za'
             * the 7zip artifact path, it will use the right 7za with your platform
             */
//            zipArtifact = "com.tencent.mm:SevenZip:1.1.10"
            /**
             * optional,default '7za'
             * you can specify the 7za path yourself, it will overwrite the zipArtifact value
             */
            /*这里改成本地安装的7za路径*/
            path = "/opt/homebrew/bin/7za"
        }

然后再次执行生成补丁任务即可。
这个问题我看其他人也遇到了,也回答了一下,参照这个也可以:https://github.com/Tencent/tinker/issues/1718

Tinker配置

官方文档和demo中的配置不是很全,毕竟文档很久没更新了,而且demo的配置比较复杂,我这里实际上用不到这么复杂的配置,下面是精简后的配置,仅供参考

/*我这里直接就用版本号来作为tinkerid了*/
def versionName = android.defaultConfig.versionName
print("versionName:" + versionName)

/*存放要生成补丁的文件夹*/
def tinkerPath = "${projectDir}/tinker/"

/**
 * 生成补丁包的配置
 * 参考官方文档:https://github.com/Tencent/tinker/wiki/Tinker-%E6%8E%A5%E5%85%A5%E6%8C%87%E5%8D%97
 */
tinkerPatch {
    
    tinkerEnable = true
    /*旧包*/
//    oldApk = "${buildDir.path}/outputs/apk/xxx.apk"
    oldApk = "${tinkerPath}/app-release.apk"
    // 补丁输出路径 选填:默认在app/build/outputs/tinkerPatch下
    outputFolder = "${tinkerPath}/patch/"
    ignoreWarning = true // 是否忽略警告
    allowLoaderInAnyDex = true // 是否支持在任意dex中加载类
    removeLoaderForAllDex = true
    useSign = true // 在运行过程中需要验证基准apk包与补丁包的签名是否一致,release包肯定是需要签名的,默认值是true
    buildConfig {
    
        /*指定旧APK的mapping文件 可减少补丁包大小*/
        applyMapping = "${tinkerPath}/mapping.txt"
        /*旧APK的resource_mapping文件 可减少补丁包大小*/
        applyResourceMapping = "${tinkerPath}/R.txt"
        /*生成tinkerid 补丁包在合并时会验证布丁包的id和基准包的id是否一致 简单点可以用versionName*/
        tinkerId = versionName
        /*是否使用加固模式,只在加固包的情况下设置为true*/
        isProtectedApp = false
        /*是否支持新增非export的activity 测试设置为true也不知道新增页面,会报错*/
        supportHotplugComponent = false
        /*如果keepDexApply为true,则dex所在的类引用旧的apk。打开这个可以减少dex-diff文件的大小。*/
        keepDexApply = false
    }

    // dex相关的配置项
    dex {
    
        /**
         * 只能是'raw'或者'jar'。 对于'raw'模式,我们将会保持输入dex的格式。对于'jar'模式,我们将会把输入dex重新压缩封装到jar。
         * 如果你的minSdkVersion小于14,你必须选择‘jar’模式,而且它更省存储空间,但是验证md5时比'raw'模式耗时。默认我们并不会去校验md5,一般情况下选择jar模式即可。
         */
        dexMode = "jar"
        /**
         * 需要处理dex路径,支持*、?通配符,必须使用'/'分割。路径是相对安装包的,例如assets/...
         * */
        pattern = ["classes*.dex", "assets/secondary-dex-?.jar"]
        // 需要处理dex路径,支持*、?通配符,必须使用'/'分割。路径是相对安装包的,例如assets/...
        /**
         * 这一项非常重要,它定义了哪些类在加载补丁包的时候会用到。这些类是通过Tinker无法修改的类,也是一定要放在main dex的类。
         * 这里需要定义的类有:
         * 1. 你自己定义的Application类;
         * 2. Tinker库中用于加载补丁包的部分类,即com.tencent.tinker.loader.*;
         * 3. 如果你自定义了TinkerLoader,需要将它以及它引用的所有类也加入loader中;
         * 4. 其他一些你不希望被更改的类,例如Sample中的BaseBuildInfo类。这里需要注意的是,这些类的直接引用类也需要加入到loader中。或者你需要将这个类变成非preverify。
         * 5. 使用1.7.6版本之后的gradle版本,参数1、2会自动填写。若使用newApk或者命令行版本编译,1、2依然需要手动填写
         * */
        loader = [
             "com.yzq.hotfix.App"//这里写你自己的Application类
        ]
    }
    //  lib相关的配置项
    lib {
    
        /**
         * 需要处理lib路径,支持*、?通配符,必须使用'/'分割。与dex.pattern一致, 路径是相对安装包的,例如assets/...
         * 一般来讲我们的so文件都放在下面两个路径下面
         */
        pattern = ["lib/*/*.so", "src/main/jniLibs/*/*.so"]
    }
    // res相关的配置项
    res {
    
        /**
         * 需要处理res路径,支持*、?通配符,必须使用'/'分割。与dex.pattern一致, 路径是相对安装包的,例如assets/...,务必注意的是,只有满足pattern的资源才会放到合成后的资源包。
         */
        pattern = ["res/*", "assets/*", "resources.arsc", "AndroidManifest.xml"]
        /**
         * 支持*、?通配符,必须使用'/'分割。若满足ignoreChange的pattern,在编译时会忽略该文件的新增、删除与修改。 最极端的情况,ignoreChange与上面的pattern一致,即会完全忽略所有资源的修改。
         */
        ignoreChange = [
                "assets/sample_meta.txt"
        ]
        largeModSize = 100
        // 对于修改的资源,如果大于largeModSize,我们将使用bsdiff算法。这可以降低补丁包的大小,但是会增加合成时的复杂度。默认大小为100kb
    }
    // 7zip路径配置项,执行前提是useSign为true
    sevenZip {
    
//        zipArtifact = "com.tencent.mm:SevenZip:1.2.17"
        /**
         * 系统中的7za路径,例如"/usr/local/bin/7za"。path设置会覆盖zipArtifact,若都不设置,将直接使用7za去尝试。
         * 这里改成本地安装的7za路径
         * */
        path = "/opt/homebrew/bin/7za"
//        path = "/Users/yuzhiqiang/.gradle/caches/modules-2/files-2.1/com.tencent.mm/SevenZip/1.2.17"
    }
}

这里说一下applyMapping和applyResourceMapping的文件从哪里来。
一般我们通过assembleRelease打完包后,可以在build文件夹中找到apk文件和mapping文件
在这里插入图片描述
R文件则是在intermediates文件夹内
在这里插入图片描述

tinker-patch-cli jar包如何获取

如果你的补丁需要通过命令行来生成,那么你就需要用到tinker-patch-cli的jar包。那这个jar包如何获取呢?
也比较简单,首先,把tinker源码拉下来,找到buildTinkerSdk这个task,执行一下

在这里插入图片描述

随后在build目录中就能获取到jar包了。
在这里插入图片描述
至于如何使用看遵循官方文档即可。
不想编译的直接下载这个 Tinker Cli Jar文件 就行,已经生成好了。

Tinker加固包的补丁如何生成

加固包的补丁生成步骤如下

  1. isProtectedApp 配置要设置为true
  2. 编译出加固前的基准包,保存好,后面生成补丁要用。
  3. 加固前的基准包进行加固,正常上应用商店,用户使用的是加固后的基准包
  4. 发现bug,进行修复。编译出加固前的新包
  5. 使用加固前的基准包加固前的新包产生补丁
  6. 测试补丁是否正常,通过测试后发布补丁

也就是说,产生补丁都是加固前的包。使用补丁是在加固后的包上使用。不要搞错了。

补丁管理规则

这里我们需要跟服务端一起制定规则,给个参考如下:

  1. 版本和补丁之间的关系是通过配置的tinkerId关联的,补丁和App的tinkerid必须一致才能正常加载,否则校验是不通过的。
  2. App一旦发布就是基准包,此时一定要保存好这个基准包(未加固前的包)以及mapping.txt和R.txt文件,后续改版本补丁的生成都是基于该基准包生成
  3. App在一个版本内可以包含多个补丁,新补丁必须要包含旧补丁已修复的代码。
  4. 补丁生效前一定要先通过测试才能下发。
  5. 服务端给补丁时一定是当前App版本所对应的最新版本的补丁。

看官方文档也能看出来,Tinker的集成以及使用还是比较繁琐的,代码量也不小,具体的代码还需要根据自己的App来做更改。

好了,本篇文章就是这样,希望能帮到你。


如果你觉得本文对你有帮助,麻烦动动手指顶一下,可以帮助到更多的开发者,如果文中有什么错误的地方,还望指正,转载请注明转自喻志强的博客 ,谢谢!

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/yuzhiqiang_1993/article/details/132761063

智能推荐

获取大于等于一个整数的最小2次幂算法(HashMap#tableSizeFor)_整数 最小的2的几次方-程序员宅基地

文章浏览阅读2w次,点赞51次,收藏33次。一、需求给定一个整数,返回大于等于该整数的最小2次幂(2的乘方)。例: 输入 输出 -1 1 1 1 3 4 9 16 15 16二、分析当遇到这个需求的时候,我们可能会很容易想到一个"笨"办法:..._整数 最小的2的几次方

Linux 中 ss 命令的使用实例_ss@,,x,, 0-程序员宅基地

文章浏览阅读865次。选项,以防止命令将 IP 地址解析为主机名。如果只想在命令的输出中显示 unix套接字 连接,可以使用。不带任何选项,用来显示已建立连接的所有套接字的列表。如果只想在命令的输出中显示 tcp 连接,可以使用。如果只想在命令的输出中显示 udp 连接,可以使用。如果不想将ip地址解析为主机名称,可以使用。如果要取消命令输出中的标题行,可以使用。如果只想显示被侦听的套接字,可以使用。如果只想显示ipv4侦听的,可以使用。如果只想显示ipv6侦听的,可以使用。_ss@,,x,, 0

conda activate qiuqiu出现不存在activate_commandnotfounderror: 'activate-程序员宅基地

文章浏览阅读568次。CommandNotFoundError: 'activate'_commandnotfounderror: 'activate

Kafka 实战 - Windows10安装Kafka_win10安装部署kafka-程序员宅基地

文章浏览阅读426次,点赞10次,收藏19次。完成以上步骤后,您已在 Windows 10 上成功安装并验证了 Apache Kafka。在生产环境中,通常会将 Kafka 与外部 ZooKeeper 集群配合使用,并考虑配置安全、监控、持久化存储等高级特性。在生产者窗口中输入一些文本消息,然后按 Enter 发送。ZooKeeper 会在新窗口中运行。在另一个命令提示符窗口中,同样切换到 Kafka 的。Kafka 服务器将在新窗口中运行。在新的命令提示符窗口中,切换到 Kafka 的。,应显示已安装的 Java 版本信息。_win10安装部署kafka

【愚公系列】2023年12月 WEBGL专题-缓冲区对象_js 缓冲数据 new float32array-程序员宅基地

文章浏览阅读1.4w次。缓冲区对象(Buffer Object)是在OpenGL中用于存储和管理数据的一种机制。缓冲区对象可以存储各种类型的数据,例如顶点、纹理坐标、颜色等。在渲染过程中,缓冲区对象中存储的数据可以被复制到渲染管线的不同阶段中,例如顶点着色器、几何着色器和片段着色器等,以完成渲染操作。相比传统的CPU访问内存,缓冲区对象的数据存储和管理更加高效,能够提高OpenGL应用的性能表现。_js 缓冲数据 new float32array

四、数学建模之图与网络模型_图论与网络优化数学建模-程序员宅基地

文章浏览阅读912次。(1)图(Graph):图是数学和计算机科学中的一个抽象概念,它由一组节点(顶点)和连接这些节点的边组成。图可以是有向的(有方向的,边有箭头表示方向)或无向的(没有方向的,边没有箭头表示方向)。图用于表示各种关系,如社交网络、电路、地图、组织结构等。(2)网络(Network):网络是一个更广泛的概念,可以包括各种不同类型的连接元素,不仅仅是图中的节点和边。网络可以包括节点、边、连接线、路由器、服务器、通信协议等多种组成部分。网络的概念在各个领域都有应用,包括计算机网络、社交网络、电力网络、交通网络等。_图论与网络优化数学建模

随便推点

android 加载布局状态封装_adnroid加载数据转圈封装全屏转圈封装-程序员宅基地

文章浏览阅读1.5k次。我们经常会碰见 正在加载中,加载出错, “暂无商品”等一系列的相似的布局,因为我们有很多请求网络数据的页面,我们不可能每一个页面都写几个“正在加载中”等布局吧,这时候将这些状态的布局封装在一起就很有必要了。我们可以将这些封装为一个自定布局,然后每次操作该自定义类的方法就行了。 首先一般来说,从服务器拉去数据之前都是“正在加载”页面, 加载成功之后“正在加载”页面消失,展示数据;如果加载失败,就展示_adnroid加载数据转圈封装全屏转圈封装

阿里云服务器(Alibaba Cloud Linux 3)安装部署Mysql8-程序员宅基地

文章浏览阅读1.6k次,点赞23次,收藏29次。PS: 如果执行sudo grep 'temporary password' /var/log/mysqld.log 后没有报错,也没有任何结果显示,说明默认密码为空,可以直接进行下一步(后面设置密码时直接填写新密码就行)。3.(可选)当操作系统为Alibaba Cloud Linux 3时,执行如下命令,安装MySQL所需的库文件。下面示例中,将创建新的MySQL账号,用于远程访问MySQL。2.依次运行以下命令,创建远程登录MySQL的账号,并允许远程主机使用该账号访问MySQL。_alibaba cloud linux 3

excel离散度图表怎么算_excel离散数据表格-Excel 离散程度分析图表如何做-程序员宅基地

文章浏览阅读7.8k次。EXCEL中数据如何做离散性分析纠错。离散不是均值抄AVEDEV……=AVEDEV(A1:A100)算出来的是A1:A100的平均数。离散是指各项目间指标袭的离散均值(各数值的波动情况),数值较低表明项目间各指标波动幅百度小,数值高表明波动幅度较大。可以用excel中的离散公式为STDEV.P(即各指标平均离散)算出最终度离散度。excel表格函数求一组离散型数据,例如,几组C25的...用exc..._excel数据分析离散

学生时期学习资源同步-JavaSE理论知识-程序员宅基地

文章浏览阅读406次,点赞7次,收藏8次。i < 5){ //第3行。int count;System.out.println ("危险!System.out.println(”真”);System.out.println(”假”);System.out.print(“姓名:”);System.out.println("无匹配");System.out.println ("安全");

linux 性能测试磁盘状态监测:iostat监控学习,包含/proc/diskstats、/proc/stat简单了解-程序员宅基地

文章浏览阅读3.6k次。背景测试到性能、压力时,经常需要查看磁盘、网络、内存、cpu的性能值这里简单介绍下各个指标的含义一般磁盘比较关注的就是磁盘的iops,读写速度以及%util(看磁盘是否忙碌)CPU一般比较关注,idle 空闲,有时候也查看wait (如果wait特别大往往是io这边已经达到了瓶颈)iostatiostat uses the files below to create ..._/proc/diskstat

glReadPixels读取保存图片全黑_glreadpixels 全黑-程序员宅基地

文章浏览阅读2.4k次。问题:在Android上使用 glReadPixel 读取当前渲染数据,在若干机型(华为P9以及魅族某魅蓝手机)上读取数据失败,glGetError()没有抓到错误,但是获取到的数据有误,如果将获取到的数据保存成为图片,得到的图片为黑色。解决方法:glReadPixels实际上是从缓冲区中读取数据,如果使用了双缓冲区,则默认是从正在显示的缓冲(即前缓冲)中读取,而绘制工作是默认绘制到后缓..._glreadpixels 全黑