技术标签: 补丁管理 android 加固包生成补丁 Tinker 热修复 Android
热修复早已不是什么新鲜技术了,各个大厂基本都有自己的热修复方案。
关于各个热修方案的对比就不赘述了,网上一搜一大堆。直接看Tinker官方文档 就行。
这里说下为什么选择Tinker吧
先说优点
说完优点再吐槽下缺点
这里我说下我自己比较推荐的集成步骤
集成步骤官方文档写的也比较清楚了,只是demo的agp版本比较老,一些配置的增加也没有在文档中更新。因此,在高版本的AGP项目中集成起来还是要踩一些坑的。
这里我把我集成tinker时踩的坑记录一下,有些坑也花了点时间去解决,希望能帮到需要的同学。
先说下我的AGP版本,这个可能会影响到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'
}
我是在M1的电脑上生成补丁包,过程中提示找不到7zip。
根据报错的信息 Could not find SevenZip-1.1.10-osx-aarch_64.exe (com.tencent.mm:SevenZip:1.1.10).
中的.exe 也能猜出来是平台相关的问题。
这里我们可以按照以下步骤做解决这个问题
brew install p7zip
本地安装下7zipwhich 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
官方文档和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包。那这个jar包如何获取呢?
也比较简单,首先,把tinker源码拉下来,找到buildTinkerSdk这个task,执行一下
随后在build目录中就能获取到jar包了。
至于如何使用看遵循官方文档即可。
不想编译的直接下载这个 Tinker Cli Jar文件 就行,已经生成好了。
加固包的补丁生成步骤如下
也就是说,产生补丁都是加固前的包。使用补丁是在加固后的包上使用。不要搞错了。
这里我们需要跟服务端一起制定规则,给个参考如下:
看官方文档也能看出来,Tinker的集成以及使用还是比较繁琐的,代码量也不小,具体的代码还需要根据自己的App来做更改。
好了,本篇文章就是这样,希望能帮到你。
如果你觉得本文对你有帮助,麻烦动动手指顶一下,可以帮助到更多的开发者,如果文中有什么错误的地方,还望指正,转载请注明转自喻志强的博客 ,谢谢!
文章浏览阅读341次。怎样优化一人四机搬运流程,不会出现workerpool手上有工件在工位前发呆,导致后工位的停止_plant工人
文章浏览阅读6k次。问题:在使用okhttp下载文件的时候拿到的文件长度为-1解决思路1.看服务器返回的请求头是否含有Content-Lengthcmd 打开命令窗口输入 curl -i +下载链接窗口显示 connect-length 是有数值的2.看返回的content-typecontent-type类型https://www.runoob.com/http/http-content-type.html第一个返回text/plain; charset=utf-8 表示文本类型..._解决okhttp content-length -1
文章浏览阅读3.4k次。首先在demo中是没有问题的,然后集成到正式项目中就报了这个错,反复对比了好几次配置都一样的(有种情况是集成了但是不使用也会报错没找到类xxx)。先Googlegoogle发现该项目的issues里面也有很多人遇到这个问题。https://github.com/HujiangTechnology/gradle_plugin_android_aspectjx/issues/101h..._aspectjtools zip file is empty
文章浏览阅读736次。原理看图说话代码实现DWORD Func_hex_to_dec(const char* szHex);int main(){ Func_hex_to_dec("F3E4"); system("pause"); return 0;}DWORD Func_hex_to_dec(const char* szHex){ DWORD dwResult = NULL; CHAR s..._vc 16进制转10进制
文章浏览阅读6.8k次,点赞3次,收藏8次。————————————————————————————————sim900GPRS模块ppp拨号上网_sim900a ppp拨号上网
文章浏览阅读149次。E10/9/8/7重写window.alert系统弹窗:文字过多自动换行,窗口右侧实现滑动精度条,倒计时弹窗确认关闭窗口
文章浏览阅读9.5k次,点赞13次,收藏30次。logPolar是把数据从笛卡尔转到对数极坐标系,linearPolar把数据从笛卡尔坐标系转到极坐标坐标系。说到笛卡尔坐标系到极坐标系的转换,大家并不陌生,公式如下:设(x, y)为极坐标的一点,()为转换中心,()为转换后的结果,则笛卡尔转到对数极坐标系的转换多了一层对数,可表示如下这里log以自然数e为底。接下来,首先分析一下logPolar和linearPola..._cv2.linearpolar
文章浏览阅读1.1k次。立足于当下餐饮行业现有的点餐模式,分析传统APP点餐的运作流程,结合Android系统的特点设计新型的外卖APP。近几年,人们生活水平日益提升,但工作强度和压力不断增强,尤其是对于上班族而言,到餐厅吃饭费时费力,而传统的APP点餐难以适应针对性。基于此,借助Web开发技术以及后台数据库,设计了以Android为平台的外卖APP系统,实现了用户登录、美食资讯、小吃商城、在线点餐、订单信息管理等功能,为人们的日常生活提供了较大的便利,同时基于Android系统的外卖APP也是信息时代发展的必然趋势。_android外卖系统主页面设计
文章浏览阅读218次。数组概述: java中用来存放多个单一数据类型的数据的容器特点: 1、未创建的时候可以存放任意数据类型的数据 2、一旦创建出来要存放的数据类型就指定了 3、一旦创建出来大小固定【长度不变】数组的初始化:【创建数组空间出来】1、静态初始化:开辟数组空间出来的时候把我们自己想要存放数据放到数组空间中,好处:定义时元素是确定的,避免内存浪费 固定格式: 标准格式: 数_数组变量储存的是什么
文章浏览阅读421次。Xmind安装默认是安装C盘。本文主要讲述如何安装Xmind在指定位置_xmind安装时无法选择目录
文章浏览阅读828次,点赞9次,收藏25次。雷达回波相位是雷达回波信号的重要特征,它反映了目标的距离和运动状态。雷达回波相位的变化规律可以分为以下几种情况:**目标静止时:**雷达回波相位保持不变。**目标向雷达靠近时:**雷达回波相位逐渐增加。**目标远离雷达时:**雷达回波相位逐渐减小。**目标横向移动时:**雷达回波相位发生周期性变化。雷达回波相位、幅度和SAR雷达回波仿真是雷达技术中的重要概念。了解这些概念对于理解雷达系统的原理和性能至关重要。_sar回波模拟
文章浏览阅读910次,点赞16次,收藏25次。基于LADRC自抗扰控制的VSG三相逆变器预同步并网控制策略是一种用于实现逆变器在微电网中的协调运行的先进控制策略。逆变器控制方式采用虚拟同步发电机控制(VSG),通过引入虚拟同步发电机的概念,为逆变器系统提供了类似于实际同步发电机的惯性和阻尼支撑。这种控制方式能够有效提高逆变器系统的稳定性和响应速度,使其更好地适应微电网的运行要求。为了增强逆变器系统的鲁邦性和抗扰能力,本模型采用了LADRC自抗扰控制。