腾讯Tinker 热修复 Andriod studio 3.0 配置和集成(三)Bugly集成
2018-01-20 15:52
656 查看
腾讯Tinker 热修复 Andriod studio 3.0 Bugly集成和多渠道补丁管理发布
本文说明上一篇我说完了腾讯Tinker 热修复之多渠道打包,这篇我们来初步了解下腾讯Tinker和Bugly结合来做热修复多渠道补丁管理和集成。(其实在上周我已经集成测试完了demo也已经上传了,就是突然想不到怎么写这篇,想了很久这篇不会写得很多,因为官方给出了视频啊,视频详细得多了,我主要分享下我所在集成中碰到的问题)
腾讯Tinker 热修复 Andriod studio 30 Bugly集成和多渠道补丁管理发布
开始
答疑
介绍
为什么使用Bugly热更新
视频教程
本文demo
Bugly配置集成
添加插件依赖
集成SDK
gradle配置
配置依赖插件脚本tinker-supportgradle
TinkerSupport插件使用指南
配置分包多渠道插件multiple-channelgradle
初始化SDK
Tinker推荐配置模式enableProxyApplication false 的情况
Tinker推荐配置模式enableProxyApplication true的情况
混淆配置
Bugly 集成后的使用
生成baseApk
生成tinker补丁包
Bugly平台使用
上传补丁
上传补丁包遇到的问题
关于碰到的问题
Bugly Android 热更新常见问题
更多
Bugly多渠道热更新解决方案
往期文章
开始
答疑
在公司做技术分享的时候,我老大提出了几个问题,我当时没有回答出来,因为确实我没看得很深入(比如源码层,文件生成目的等)这里我重新看了下官方文档解析下1.生成baseApk文件夹中的mapping.txt是什么 有什么用? R文件呢?
mapping.txt其实就是apk的混淆后的代码文本 R文件也是一样 防止反编译
2.baseApk 每次都会生成怎么管理?
baseApk对应的是上一次的oldApk,第一次是baseApk是为空的,建议自己创文件保存,根据文件名日期去管理,而补丁包也是一样。
3.TinkerSupport 插件中 tinkerPatch配置 有什么用?
它其实就是全局信息相关的配置项,对应overrideTinkerPatchConfiguration 默认flase 不启用使用默认的配置 开启可以自定义配置。
介绍
热更新能力是Bugly为解决开发者紧急修复线上bug,而无需重新发版让用户无感知就能把问题修复的一项能力。Bugly目前采用微信Tinker的开源方案,开发者只需要集成我们提供的SDK就可以实现自动下载补丁包、合成、并应用补丁的功能,我们也提供了热更新管理后台让开发者对每个版本补丁进行管理。为什么使用Bugly热更新?
无需关注Tinker是如何合成补丁的无需自己搭建补丁管理后台
无需考虑后台下发补丁策略的任何事情
无需考虑补丁下载合成的时机,处理后台下发的策略
提供了更加方便集成Tinker的方式
通过HTTPS及签名校验等机制保障补丁下发的安全性
丰富的下发维度控制,有效控制补丁影响范围
提供了应用升级一站式解决方案
视频教程
2017年3月的有参考意义
视频地址
本文demo
本文最新Bugly热修复集成包括多渠道demoBugly配置集成
添加插件依赖
andriod studio 3.0 配置 工程根目录下“build.gradle”文件中添加:
classpath 'com.android.tools.build:gradle:3.0.1' // NOTE: Do not place your application dependencies here; they belong // in the individual module build.gradle files // tinkersupport插件, 其中lastest.release指拉取最新版本,也可以指定明确版本号,例如1.0.9 // classpath "com.tencent.bugly:tinker-support:lastest.release" classpath "com.tencent.bugly:tinker-support:1.1.1"//tinker版本至1.9.1 对应bugly 1.1.1 看更新文档 // 多渠道插件(多渠道打包推荐使用) classpath 'com.meituan.android.walle:plugin:1.1.3' }
集成SDK
gradle配置
在app module的“build.gradle”文件中添加(示例配置): apply plugin: 'com.android.application' android { signingConfigs { debug { storeFile file('./keystore/debug.keystore') } release { keyAlias 'buglyrelease' keyPassword '123456' storeFile file('D:/Users/Achers/AsBuglyTinker/app/keystore/buglyrelease.jks') storePassword '123456' } } compileSdkVersion 26 defaultConfig { applicationId "com.achers.asbuglytinker" minSdkVersion 15 targetSdkVersion 26 versionCode 1 versionName "1.0" testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner" // 开启multidex multiDexEnabled true } // 编译选项 compileOptions { sourceCompatibility JavaVersion.VERSION_1_7 targetCompatibility JavaVersion.VERSION_1_7 } // recommend dexOptions { jumboMode = true } // 构建类型 buildTypes { release { minifyEnabled true proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' signingConfig signingConfigs.release } debug { debuggable true minifyEnabled false signingConfig signingConfigs.debug } } sourceSets { main { jniLibs.srcDirs = ['libs'] } } repositories { flatDir { dirs 'libs' } } // 多渠道配置 /*productFlavors { xiaomi { 123ea } yyb { } }*/ lintOptions { checkReleaseBuilds false abortOnError false } } dependencies { implementation fileTree(include: ['*.jar'], dir: 'libs') implementation 'com.android.support:appcompat-v7:26.1.0' implementation 'com.android.support.constraint:constraint-layout:1.0.2' testImplementation 'junit:junit:4.12' androidTestImplementation 'com.android.support.test:runner:1.0.1' androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.1' // 多dex配置 implementation 'com.android.support:multidex:1.0.1' // 集成Bugly热更新aar(本地集成使用方式) // compile(name: 'bugly_crashreport_upgrade-1.3.2', ext: 'aar') // 远程仓库集成方式(推荐) implementation 'com.tencent.bugly:crashreport_upgrade:1.3.4' // walle(多渠道使用) compile 'com.meituan.android.walle:library:1.1.3' } // 依赖插件脚本 apply from: 'tinker-support.gradle' // 多渠道使用walle示例(注:多渠道使用) apply from: 'multiple-channel.gradle'
配置依赖插件脚本tinker-support.gradle
在app下新建tinker-support.gradleapply plugin: 'com.tencent.bugly.tinker-support' def bakPath = file("${buildDir}/bakApk/") /** * 此处填写每次构建生成的基准包目录 */ def baseApkDir = "app-0114-18-17-58" /** * 对于插件各参数的详细解析请参考 */ tinkerSupport { // 开启tinker-support插件,默认值true enable = true // 指定归档目录,默认值当前module的子目录tinker autoBackupApkDir = "${bakPath}" autoGenerateTinkerId = false // 是否启用覆盖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 = "patch-1.0.2" // buildAllFlavorsDir = "${bakPath}/${baseApkDir}" // 是否开启加固模式,默认为false // isProtectedApp = true //默认为false 就是需要自己改造application 为true 就是不需要改造 通过反射 enableProxyApplication = false // 是否支持新增非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-patch" // applyMapping = "${bakPath}/${appName}/app-release-mapping.txt" // 可选,设置mapping文件,建议保持旧apk的proguard混淆方式 //applyResourceMapping = "${bakPath}/${appName}/app-release-R.txt" // 可选,设置R.txt文件,通过旧apk文件保持ResId的分配 } }
TinkerSupport插件使用指南
参数介绍地址TinkerSupport插件使用指南配置分包多渠道插件multiple-channel.gradle
在app下新建multiple-channel.gradleapply plugin: 'walle' walle { // 指定渠道包的输出路径 apkOutputFolder = new File("${project.buildDir}/outputs/channels"); // 定制渠道包的APK的文件名称 apkFileNameFormat = '${appName}-${packageName}-${channel}-${buildType}-v${versionName}-${versionCode}-${buildTime}.apk'; // 渠道配置文件 channelFile = new File("${project.getProjectDir()}/channel") }
初始化SDK
Tinker推荐配置模式enableProxyApplication = false 的情况
自定义Application/** * Create on 2018/1/14 12:23 * <p> * author lhm * <p> * Description: enableProxyApplication = false 的情况 * 这是Tinker推荐的接入方式,一定程度上会增加接入成本,但具有更好的兼容性。 * <p> * Version: 1.2.3 * * 注意:这个类集成TinkerApplication类,这里面不做任何操作,所有Application的代码都会放到ApplicationLike继承类当中 参数解析 参数1:tinkerFlags 表示Tinker支持的类型 dex only、library only or all suuport,default: TINKER_ENABLE_ALL 参数2:delegateClassName Application代理类 这里填写你自定义的ApplicationLike 参数3:loaderClassName Tinker的加载器,使用默认即可 参数4:tinkerLoadVerifyFlag 加载dex或者lib是否验证md5,默认为false */ public class APP extends TinkerApplication { public APP() { super(ShareConstants.TINKER_ENABLE_ALL, "com.achers.asbuglytinker.SampleApplicationLike", "com.tencent.tinker.loader.TinkerLoader", false); } }
清单文件配置 application
<application android:name=".APP" android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name" android:roundIcon="@mipmap/ic_launcher_round" android:supportsRtl="true" android:theme="@style/AppTheme"> <activity android:name=".MainActivity"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity>
SampleApplicationLike 配置
package com.achers.asbuglytinker; import android.annotation.TargetApi; import android.app.Application; import android.content.Context; import android.content.Intent; import android.os.Build; import android.support.multidex.MultiDex; import android.widget.Toast; import com.meituan.android.walle.WalleChannelReader; import com.tencent.bugly.Bugly; import com.tencent.bugly.beta.Beta; import com.tencent.bugly.beta.interfaces.BetaPatchListener; import com.tencent.tinker.loader.app.DefaultApplicationLike; import java.util.Locale; /** * Create on 2018/1/14 12:26 * <p> * author lhm * <p> * Description: * <p> * Version: 1.2.3 */ public class SampleApplicationLike extends DefaultApplicationLike { public static final String TAG = "Tinker.SampleApplicationLike"; public SampleApplicationLike(Application application, int tinkerFlags, boolean tinkerLoadVerifyFlag, long applicationStartElapsedTime, long applicationStartMillisTime, Intent tinkerResultIntent) { super(application, tinkerFlags, tinkerLoadVerifyFlag, applicationStartElapsedTime, applicationStartMillisTime, tinkerResultIntent); } @Override public void onCreate() { super.onCreate(); // 设置是否开启热更新能力,默认为true Beta.enableHotfix = true; // 设置是否自动下载补丁,默认为true Beta.canAutoDownloadPatch = true; // 设置是否自动合成补丁,默认为true Beta.canAutoPatch = true; // 设置是否提示用户重启,默认为false Beta.canNotifyUserRestart = true; // 补丁回调接口 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(), "补丁下载失败", Toast.LENGTH_SHORT).show(); } @Override public void onApplySuccess(String msg) { Toast.makeText(getApplication(), "补丁应用成功", Toast.LENGTH_SHORT).show(); } @Override public void onApplyFailure(String msg) { Toast.makeText(getApplication(), "补丁应用失败", Toast.LENGTH_SHORT).show(); } @Override public void onPatchRollback() { } }; // 设置开发设备,默认为false,上传补丁如果下发范围指定为“开发设备”,需要调用此接口来标识开发设备 Bugly.setIsDevelopmentDevice(getApplication(), true); // 多渠道需求塞入 String channel = WalleChannelReader.getChannel(getApplication()); Bugly.setAppChannel(getApplication(), channel); // 这里实现SDK初始化,appId替换成你的在Bugly平台申请的appId Bugly.init(getApplication(), "4694167734", true); } @TargetApi(Build.VERSION_CODES.ICE_CREAM_SANDWICH) @Override public void onBaseContextAttached(Context base) { super.onBaseContextAttached(base); // you must install multiDex whatever tinker is installed! MultiDex.install(base); // TODO: 安装tinker Beta.installTinker(this); } @TargetApi(Build.VERSION_CODES.ICE_CREAM_SANDWICH) public void registerActivityLifecycleCallback( Application.ActivityLifecycleCallbacks callbacks) { getApplication().registerActivityLifecycleCallbacks(callbacks); } @Override public void onTerminate() { super.onTerminate(); Beta.unInit(); } }
Tinker推荐配置模式enableProxyApplication = true的情况
无须你改造Application,主要是为了降低接入成本,我们插件会动态替换AndroidMinifest文件中的Application为我们定义好用于反射真实Application的类(需要您接入SDK 1.2.2版本 和 插件版本 1.0.3以上)。
public class MyApplication extends Application { @Override public void onCreate() { super.onCreate(); // 这里实现SDK初始化,appId替换成你的在Bugly平台申请的appId // 调试时,将第三个参数改为true Bugly.init(this, "900029763", false); } @Override protected void attachBaseContext(Context base) { super.attachBaseContext(base); // you must install multiDex whatever tinker is installed! MultiDex.install(base); // 安装tinker Beta.installTinker(); } }
混淆配置
为了避免混淆SDK,在Proguard混淆文件中增加以下配置:-dontwarn com.tencent.bugly.** -keep public class com.tencent.bugly.**{*;}
如果你使用了support-v4包,你还需要配置以下混淆规则:
-keep class android.support.**{*;}
Bugly 集成后的使用
生成baseApk
看过前两篇关于Tinker 这种基本操作就很简单了,差不多一毛一样。
查看生成目录
生成tinker补丁包
这里和tinker不同 tinker是执行 tinker包下的 而bugly是执行tinker-support
生成补丁包的同时会生成新的baseApk文件夹 这个就是在oldApk(有bug)基础上更新代码后执行tinker-support NewApk(修复版)上图
Bugly平台使用
上传补丁
选择项目文件下的7zip 注意不是thinkePatch下的apk 而是 patch下的
上传补丁包遇到的问题
原因是因为我们的baseApk虽然集成了bugly 但是没有在运行就是没有联网与bugly对接 平台没有记录,所以找不到。解决办法 运行下baseApk 就可以。
关于碰到的问题
tinkerid的设置
这里官方是默认为true的 它会自己生成tinkerid 格式是versionName+versionCode+生成时间,他的意思很明显就是你不配置tinkerid 就是按这个格式来配置,你就不用自己每次改一变tinkerid 但是非常难看 比如 1.01TinkerId2018-14-11-20;看起来很难管理 因为Bugly与tinker不同的地方在与他会在补丁包中增加一个文件YAPATCH.MF用来记录补丁
这是我将autoGenerateTinkerId=false 后 每次手动设置的tinkerid tinkerid 规则 基础包为 base-1.0.1 对应补丁包 patch-1.0.1 基础包为 base-1.0.2 对应补丁包 patch-1.0.2 就是为了自己定义方便管理。
Bugly Android 热更新常见问题
Q:之前使用Tinker怎么切换过来使用Bugly?A:Bugly使用源码集成Tinker,如果之前集成过Tinker,你需要注释掉以下配置:
// compile("com.tencent.tinker:tinker-android-lib:${TINKER_VERSION}") { changing = true } // provided("com.tencent.tinker:tinker-android-anno:${TINKER_VERSION}") { changing = true }
插件配置不需要更改,只需要加上我们Bugly额外的tinker-support插件即可:
classpath "com.tencent.bugly:tinker-support:latest.release"
Q: 基线版本表示什么意思?
A:表示你需要修复apk的版本,比如你已经上线了某个版本的apk,你需要用一个唯一的tinkerId来标识这个版本,而补丁包也是基于这个版本打的。
Q:打一个补丁包需要改哪些东西?
A:
修复bug的类、修改资源
修改oldApk配置
修改tinkerId
Q:tinkerId该怎么填?
A:在运行过程中,我们需要验证基准apk包的tinkerId是否等于补丁包的tinkerId。这个是决定补丁包能运行在哪些基准包上面,一般来说我们可以使用git版本号、versionName等等。
Q:两次传入的tinkerId是否一样?
A:不一样的,编译补丁包时,tinker会自动读取基准包AndroidManifest的tinkerId作为package_meta.txt中的TINKER_ID。将本次编译传入的tinkerId,作为package_meta.txt的NEW_TINKER_ID。
Q. 为什么我上传补丁提示我“未匹配到可用补丁的App版本”?
A:如果你的基线版本没有上报过联网,基于这个版本生成的补丁包就无法匹配到,请检查你的基线版本配置是否正确。(具体参考:启动apk,上报联网数据)
Q. 我该上传哪个补丁?patch目录跟tinkerPatch目录下的补丁有什么区别吗?
A:你必须要上传build/outputs/patch目录下的补丁包,
Q:我以前的是Bugly SDK,现在集成升级SDK会有什么影响?
A:不会有影响的,升级SDK内置Bugly功能模块,你只需要将初始化方法改为统一的Bugly.init(getApplicationContext(), “注册时申请的APPID”, false);即可。
Q:你们是怎么定义开发设备的?
A:我们会提供接口Bugly.setIsDevelopmentDevice(getApplicationContext(), true);,我们后台就会将你当前设备识别为开发设备,如果设置为false则非开发设备,我们会根据这个配置进行策略控制。
Q:如果我配置了升级策略,又配置了补丁策略,会是怎样的效果?
A:升级策略优先级会高于补丁策略,后台会优先下发升级策略。毕竟你都要升级了,热更新只是帮助你修复bug而已。
Q:我只想使用热更新,不想使用升级?
A:热更新是包含在升级SDK里面的,你可以不配置任何升级策略,只需按照热更新文档集成即可。
Q:是否支持加固模式?
A:
Bugly 1.3.0及以上版本支持(tinker 1.7.9)
需要你在tinker-support配置中设置isProtectedApp = true。
更多
热更新常见问题Bugly多渠道热更新解决方案
bugly多渠道就跟tinker一毛一样了,这里贴这些没什么意义大家请看官方文章
Bugly多渠道热更新解决方案
往期文章
腾讯Tinker 热修复 Andriod studio 3.0 配置和集成腾讯Tinker 热修复 Andriod studio 3.0 多渠道打包和发布补丁方式推荐
相关文章推荐
- 腾讯Tinker 热修复 Andriod studio 3.0 配置和集成(一)
- 腾讯Tinker 热修复 Andriod studio 3.0 配置和集成(二)多渠道打包和补丁发布
- 关于使用腾讯 Bugly 平台 Tinker开源热修复框架的 项目集成
- Android studio3.0 命令行方式简单集成腾讯Tinker热修复插件
- 热修复框架Tinker的从0到集成之路
- 热修复框架Tinker快速集成
- Enterprise Library 3.0 体验(1):集成VS2005的配置工具和配置区加密支持
- 微信Tinker热修复集成
- Android 三分钟集成微信热修复Tinker
- Enterprise Library 3.0 体验(1):集成VS2005的配置工具和配置区加密支持
- Android Tinker热修复集成解析
- 【Android】微信热修复 Tinker 的集成和使用
- Enterprise Library 3.0 体验(1):集成VS2005的配置工具和配置区加密支持
- 关于微信Tinker热修复自己的集成方式以及问题
- Android探索之旅(第二十三篇)Android中集成Tinker热修复及所碰到的坑
- Tinker热修复,配置时的坑
- CKEditor3.0在asp.net环境下上传文件的配置,集成CKFinder
- [Android]腾讯Tinker热修复框架简单使用
- 热修复框架Tinker的从0到集成之路
- 热修复——Tinker 的集成与使用