先说下我遇到的一些问题:
tinker1.9.0开始支持修改清单文件,可新增activity(exported = false)应用内部访问的activity;暂不支持补丁更改应用图标;
release版本包混淆后, 当打7zip补丁包的时候,可能会报出一些三方,如地图api的一些警告;
举个栗子:
修改app proguard-rules.pro文件,增加如下忽略,解决;
# 导航
-dontwarn com.amap.**
配置:
1.相关配置:
【1】project build.gradle:(gradle3.0.0需要配置google())
repositories {
google()
jcenter()
}
dependencies {
classpath 'com.android.tools.build:gradle:3.0.0'
// tinker-support 1.0.9 对应 tinker 1.9.0
// tinker-support 1.1.1 对应 tinker 1.9.1
classpath "com.tencent.bugly:tinker-support:1.1.1"
}
【2】
app gradle :
// 远程仓库集成方式(推荐) 热修复
compile 'com.tencent.bugly:crashreport_upgrade:1.3.4'
//类似分包机制解决方案,64k解决方案
compile 'com.android.support:multidex:1.0.1'
// 依赖插件脚本
apply from: 'tinker-support.gradle'
若项目中引入dagger2或者butterknife等三方插件,需要apt工具编译,需要修改相关配置;
//gradle3.0.0的构建方式
annotationProcessor 'com.google.dagger:dagger-compiler:2.6'
annotationProcessor 'com.jakewharton:butterknife-compiler:8.2.1'
签名配置:
signingConfigs {
release {
keyAlias '******'
keyPassword '***'
storePassword '*********'
storeFile file('D:/keystore/logistics.jks')
}
}
代码混淆:release版本的包,必须混淆;否则编译无法生成mapping文件
minifyEnabled true
【3】Manifest清单文件 , 若provider 里有用到,那么,直接增加到自己的provider中;
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths"/>
res下的xml文件夹下的file_paths.xml中加入:
若没有,则直接复制以下
android:name="com.tencent.bugly.beta.ui.BetaActivity"
android:configChanges="keyboardHidden|orientation|screenSize|locale"
android:theme="@android:style/Theme.Translucent"/>
android:name="android.support.v4.content.FileProvider"
android:authorities="${applicationId}.fileProvider"
android:exported="false"
android:grantUriPermissions="true">
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/provider_paths"/>
【4】app module 创建 tinker-support.gradle 文件
apply plugin: 'com.tencent.bugly.tinker-support'
def bakPath = file("${buildDir}/bakApk/")
/**
*此处填写每次构建生成的基准包目录,用户安装的版本,用于对比生成差分包的目录;
*/
def baseApkDir = "app-1220-11-39-55"
/**
* 对于插件各参数的详细解析请参考
*/
tinkerSupport {
// 开启tinker-support插件,默认值true
enable = true
// 指定归档目录,默认值当前module的子目录tinker
autoBackupApkDir = "${bakPath}"
// 是否启用覆盖tinkerPatch配置功能,默认值false
// 开启后tinkerPatch配置不生效,即无需添加tinkerPatch
overrideTinkerPatchConfiguration = true
// 编译补丁包时,必需指定基线版本的apk,默认值为空
// 如果为空,则表示不是进行补丁包的编译
// @{link tinkerPatch.oldApk }
baseApk = "${bakPath}/${baseApkDir}/app-release.apk"
// 对应tinker插件applyMapping
baseApkProguardMapping = "${bakPath}/${baseApkDir}/app-release-mapping.txt"
// 对应tinker插件applyResourceMapping
baseApkResourceMapping = "${bakPath}/${baseApkDir}/app-release-R.txt"
// 构建基准包和补丁包都要指定不同的tinkerId,并且必须保证唯一性
tinkerId = "patch-1.3.0"
// 构建多渠道补丁时使用
// buildAllFlavorsDir = "${bakPath}/${baseApkDir}"
// 是否启用加固模式,默认为false.(tinker-spport 1.0.7起支持)
// isProtectedApp = true
// 是否开启反射Application模式 , 这种模式不需要更改application结构,但是兼容性不高;
enableProxyApplication = true
// 是否支持新增非export的Activity(注意:设置为true才能修改AndroidManifest文件)
supportHotplugComponent = true
}
/**
* 一般来说,我们无需对下面的参数做任何的修改
* 对于各参数的详细介绍请参考:
* https://github.com/Tencent/tinker/wiki/Tinker-%E6%8E%A5%E5%85%A5%E6%8C%87%E5%8D%97
*/
tinkerPatch {
//oldApk ="${bakPath}/${appName}/app-release.apk"
ignoreWarning = false
useSign = true
dex {
dexMode = "jar"
pattern = ["classes*.dex"]
loader = []
}
lib {
pattern = ["lib/*/*.so"]
}
res {
pattern = ["res/*", "r/*", "assets/*", "resources.arsc", "AndroidManifest.xml"]
ignoreChange = []
largeModSize = 100
}
packageConfig {
}
sevenZip {
zipArtifact = "com.tencent.mm:SevenZip:1.1.10"
// path = "/usr/local/bin/7za"
}
buildConfig {
keepDexApply = false
//tinkerId = "1.0.1-base"
//applyMapping = "${bakPath}/${appName}/app-release-mapping.txt" // 可选,设置mapping文件,建议保持旧apk的proguard混淆方式
//applyResourceMapping = "${bakPath}/${appName}/app-release-R.txt" // 可选,设置R.txt文件,通过旧apk文件保持ResId的分配
}
}
【5】application--反射形式配置:
平台注册appid ;
@Override
public void onCreate() {
super.onCreate ();
//bugly热修复 //若设置称true,只要手机连接电脑,就会在logcat中打印相关操作日志,日志Crash过滤;
Bugly.init (this, BUGLY_APPID, false);
}
@Override
protected void attachBaseContext(Context base) {
super.attachBaseContext (base);
MultiDex.install (this);
// 安装tinker
Beta.installTinker ();
//===========以下配置建议添加,亲测若不添加,vivo手机补丁不生效;
// 设置是否开启热更新能力,默认为true
Beta.enableHotfix = true;
// 设置是否自动下载补丁,默认为true
Beta.canAutoDownloadPatch = true;
// 设置是否自动合成补丁,默认为true
Beta.canAutoPatch = true;
// 设置是否提示用户重启,默认为false
Beta.canNotifyUserRestart = false;
// 补丁回调接口
Beta.betaPatchListener = new BetaPatchListener () {
@Override
public void onPatchReceived(String patchFile) {
// Toast.makeText (getApplication (), "补丁下载地址" + patchFile, Toast.LENGTH_SHORT).show ();
}
@Override
public void onDownloadReceived(long savedLength, long totalLength) {
/* Toast.makeText (getApplication (),
String.format (Locale.getDefault (), "%s %d%%",
Beta.strNotificationDownloading,
(int) (totalLength == 0 ? 0 : savedLength * 100 / totalLength)),
Toast.LENGTH_SHORT).show ();*/
}
@Override
public void onDownloadSuccess(String msg) {
// Toast.makeText (getApplication (), "补丁下载成功", Toast.LENGTH_SHORT).show ();
}
@Override
public void onDownloadFailure(String msg) {
// Toast.makeText (getApplication (), "补丁下载失败" + msg, Toast.LENGTH_SHORT).show ();
}
@Override
public void onApplySuccess(String msg) {
// Toast.makeText (getApplication (), "补丁应用成功" + msg, Toast.LENGTH_SHORT).show ();
}
@Override
public void onApplyFailure(String msg) {
Toast.makeText (getApplication (), "补丁应用失败" + "========" + msg, Toast.LENGTH_SHORT).show ();
}
@Override
public void onPatchRollback() {
}
};
}
============================配置完成===========================请开始你的表演===================================
1.生成基准包,用户安装的基准包,需要本地将这个bakApk包保存,用于线上出现bug,对比生成补丁包;
双击后,会在app下生成bakApk的文件夹;
这个app-release包,用于用户安装使用;(必须保证联网使用了,否则上传补丁包的时候,网页会报,未匹配到使用app的用户信息)
查看apk中,发现目前我们的tinkerId版本是1.3.0;
2.修改相应bug,然后生成我们的补丁包;
【1】修改tinker-support。gradle中的相关信息;
【2】增加tinkerId的版本号;
// 构建基准包和补丁包都要指定不同的tinkerId,并且必须保证唯一性,建议使用VersionCode或者VersionName作为名称;
tinkerId = "patch-1.4.0"
【3】修复相应bug
【4】生成补丁包;
双击:
outputs中生成patch文件夹;
这个7Zip.apk就是生成的补丁包;
查看里面的.MF文件,这里可以看到我们的TinkerId从1.3版本,更新到了1.4版本;
【5】开始上传补丁啦; 选择全量设备;也可以自定义,设定升级的android版本范围;4x 5x以上等; 开发设备用于开发者自己测试使用;后面会讲到;
下发完成后,会等待下次进入应用的时候,修复相应bug;但是需要1-10分钟的时间;
下发的补丁也可以撤回,撤回后相当于未安装该补丁;
==============================================完成=========================================================
最后说下多渠道打包:
多渠道打包的时候,tinker-support.gradle中必须加上这个;
// 构建多渠道补丁时使用
buildAllFlavorsDir = "${bakPath}/${baseApkDir}"
打补丁包的时候,选择生成全部的差分包
挨个发布多个补丁包即可;