TinkerTest
演示如何使用騰訊的熱修復(fù)框架-Tinker
Tinker熱更新演示(請(qǐng)star支持)
演示demo下載
Tinker簡(jiǎn)介
Tinker是微信官方的Android熱補(bǔ)丁解決方案,它支持動(dòng)態(tài)下發(fā)代碼、So庫以及資源,讓應(yīng)用能夠在不需要重新安裝的情況下實(shí)現(xiàn)更新。當(dāng)然,你也可以使用Tinker來更新你的插件。
相關(guān)鏈接
Tinker已知問題
由于原理與系統(tǒng)限制,Tinker有以下已知問題:
- Tinker不支持修改AndroidManifest.xml,Tinker不支持新增四大組件(1.9.0支持新增非export的Activity);
- 由于Google Play的開發(fā)者條款限制,不建議在GP渠道動(dòng)態(tài)更新代碼;
- 在Android N上,補(bǔ)丁對(duì)應(yīng)用啟動(dòng)時(shí)間有輕微的影響;
- 不支持部分三星android-21機(jī)型,加載補(bǔ)丁時(shí)會(huì)主動(dòng)拋出"TinkerRuntimeException:checkDexInstall failed";
- 對(duì)于資源替換,不支持修改remoteView。例如transition動(dòng)畫,notification icon以及桌面圖標(biāo)。
官方說明請(qǐng)點(diǎn)擊查看.
Tinker接入
添加依賴
- 在Project的根目錄的build.gradle下添加
tinkerpatch插件:
buildscript {
...
dependencies {
...
classpath "com.tinkerpatch.sdk:tinkerpatch-gradle-plugin:1.2.8"
}
}
- 在module的build.gradle下增加
Tinker的依賴。
dependencies {
implementation 'com.android.support:multidex:1.0.3'
//若使用annotation需要單獨(dú)引用,對(duì)于tinker的其他庫都無需再引用
annotationProcessor 'com.tinkerpatch.tinker:tinker-android-anno:1.9.8'
compileOnly 'com.tinkerpatch.tinker:tinker-android-anno:1.9.8'
implementation 'com.tinkerpatch.sdk:tinkerpatch-android-sdk:1.2.8'
}
- 配置代碼混淆和打包配置。其中tinkerMultidexKeep.pro和proguardRules.pro可參考我的demo工程。
android {
compileSdkVersion 27
defaultConfig {
applicationId "com.xuexiang.tinkertest"
minSdkVersion 14
targetSdkVersion 27
versionCode 1
versionName "1.0"
multiDexEnabled true
multiDexKeepProguard file("tinkerMultidexKeep.pro") //keep specific classes using proguard syntax
}
signingConfigs {
release {
//配置你的storekey
storeFile file(app_release.storeFile)
storePassword app_release.storePassword
keyAlias app_release.keyAlias
keyPassword app_release.keyPassword
}
}
buildTypes {
release {
minifyEnabled true
shrinkResources true
signingConfig signingConfigs.release
proguardFiles 'proguardRules.pro', getDefaultProguardFile('proguard-android.txt')
}
debug {
debuggable true
minifyEnabled false
}
}
sourceSets {
main {
jniLibs.srcDirs = ['libs']
}
}
}
- 引用
tinkerpatch.gradle執(zhí)行腳本.
apply from: 'tinkerpatch.gradle'
- 配置
tinkerpatch.gradle執(zhí)行腳本.對(duì)于腳本的詳細(xì)說明請(qǐng)參考官方文檔。
這里我只想說幾個(gè)比較關(guān)鍵的配置:
bakPath: 這里存放的是每次我們打包生成的apk目錄,作為生成補(bǔ)丁包的參考包路徑。(一般配置了就不用動(dòng)了)
baseInfo: 這里設(shè)置的是,本次打補(bǔ)丁包參照的apk包所在文件夾路徑,也就是舊APK的目錄。(每次打補(bǔ)丁包的時(shí)候,都需要手動(dòng)去更新)
variantName: 設(shè)置打補(bǔ)丁包的類型是release還是debug。
AppKey: 這是從Tinker Platform上注冊(cè)獲得的應(yīng)用的appkey。
AppVersion: 這也是在Tinker Platform上,每次上傳補(bǔ)丁包時(shí)都需要填寫的應(yīng)用版本,并且必須是唯一的。
【注意】:AppKey和AppVersion都是用于Tinker Platform自定發(fā)布補(bǔ)丁包所需要的。如果你不使用Tinker Platform來管理你的熱更新的話,可以隨便設(shè)置。
以下是tinkerpatch.gradle的配置樣例:
apply plugin: 'tinkerpatch-support'
/**
* TODO: 請(qǐng)按自己的需求修改為適應(yīng)自己工程的參數(shù)
*/
def bakPath = file("${buildDir}/bakApk/")
/** 每次在打補(bǔ)丁包的時(shí)候,需要更新這里的舊包的位置 **/
def baseInfo = "app-1.0.0-0810-17-28-31"
def variantName = "release"
def AppKey = "4c118de195c79b14"
def AppVersion = "1.0.0"
/**
* 對(duì)于插件各參數(shù)的詳細(xì)解析請(qǐng)參考
* http://tinkerpatch.com/Docs/SDK
*/
tinkerpatchSupport {
/** 可以在debug的時(shí)候關(guān)閉 tinkerPatch **/
/** 當(dāng)disable tinker的時(shí)候需要添加multiDexKeepProguard和proguardFiles,
這些配置文件本身由tinkerPatch的插件自動(dòng)添加,當(dāng)你disable后需要手動(dòng)添加
你可以copy本示例中的proguardRules.pro和tinkerMultidexKeep.pro,
需要你手動(dòng)修改'tinker.sample.android.app'本示例的包名為你自己的包名, com.xxx前綴的包名不用修改
**/
tinkerEnable = true
/** 是否使用一鍵接入功能 **/
reflectApplication = true
/**
* 是否開啟加固模式,只能在APK將要進(jìn)行加固時(shí)使用,否則會(huì)patch失敗。
* 如果只在某個(gè)渠道使用了加固,可使用多flavors配置
**/
protectedApp = false
/**
* 實(shí)驗(yàn)功能
* 補(bǔ)丁是否支持新增 Activity (新增Activity的exported屬性必須為false)
**/
supportComponent = true
/** 在tinkerpatch.com得到的appKey,改成你的應(yīng)用appKey **/
appKey = "${AppKey}"
/** 注意: 若發(fā)布新的全量包, appVersion一定要更新 **/
appVersion = "${AppVersion}"
autoBackupApkPath = "${bakPath}"
def pathPrefix = "${bakPath}/${baseInfo}/${variantName}/"
def name = "${project.name}-${variantName}"
baseApkFile = "${pathPrefix}/${name}.apk"
baseProguardMappingFile = "${pathPrefix}/${name}-mapping.txt"
baseResourceRFile = "${pathPrefix}/${name}-R.txt"
/**
* 若有編譯多flavors需求, 可以參照: https://github.com/TinkerPatch/tinkerpatch-flavors-sample
* 注意: 除非你不同的flavor代碼是不一樣的,不然建議采用zip comment或者文件方式生成渠道信息(相關(guān)工具:walle 或者 packer-ng)
**/
}
/**
* 用于用戶在代碼中判斷tinkerPatch是否被使能
*/
android {
defaultConfig {
buildConfigField "boolean", "TINKER_ENABLE", "${tinkerpatchSupport.tinkerEnable}"
}
}
/**
* 一般來說,我們無需對(duì)下面的參數(shù)做任何的修改
* 對(duì)于各參數(shù)的詳細(xì)介紹請(qǐng)參考:
* https://github.com/Tencent/tinker/wiki/Tinker-%E6%8E%A5%E5%85%A5%E6%8C%87%E5%8D%97
*/
tinkerPatch {
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
}
}
- 最后就是配置Application了。因?yàn)槲以谏厦嬖O(shè)置
reflectApplication = true使用了一鍵接入功能,所以就不需要進(jìn)行復(fù)雜的配置了,如下:
public class App extends Application {
@Override
public void onCreate() {
super.onCreate();
// 我們可以從這里獲得Tinker加載過程的信息
ApplicationLike tinkerApplicationLike = TinkerPatchApplicationLike.getTinkerPatchApplicationLike();
// 初始化TinkerPatch SDK, 更多配置可參照API章節(jié)中的,初始化SDK
TinkerPatch.init(tinkerApplicationLike)
.reflectPatchLibrary()
.setPatchRollbackOnScreenOff(true)
.setPatchRestartOnSrceenOff(true)
.setPatchResultCallback(new ResultCallBack() {
@Override
public void onPatchResult(PatchResult patchResult) {
ToastUtils.toast("補(bǔ)丁修復(fù):" + (patchResult.isSuccess ? "成功" : "失敗"));
}
})
.setFetchPatchIntervalByHours(3);
// 每隔3個(gè)小時(shí)(通過setFetchPatchIntervalByHours設(shè)置)去訪問后臺(tái)時(shí)候有更新,通過handler實(shí)現(xiàn)輪訓(xùn)的效果
TinkerPatch.with().fetchPatchUpdateAndPollWithInterval();
}
}
更多復(fù)雜的配置和高端自定義操作可參見官方文檔。
以上就完成了Tinker的接入工作。
如何使用Tinker進(jìn)行熱修復(fù)
- 先隨便在程序中寫一個(gè)bug,然后執(zhí)行
./gradlew assembleRelease進(jìn)行打包。當(dāng)然你也可以直接在AS右側(cè)的Gradle中找到你的應(yīng)用,并在Tasks->build->assembleRelease找到assembleRelease的任務(wù),雙擊執(zhí)行任務(wù)。
執(zhí)行完成后,你會(huì)在你模塊的build->bakApk下看到你打的apk包。
你將剛才生成apk的那個(gè)文件夾的名稱設(shè)置在之前說的
tinkerpatch.gradle中的baseInfo。將bug修復(fù)后,執(zhí)行
./gradlew tinkerPatchRelease打補(bǔ)丁包。當(dāng)然你也可以直接在AS右側(cè)的Gradle中找到你的應(yīng)用,并在Tasks->tinker->tinkerPatchRelease找到tinkerPatchRelease的任務(wù),雙擊執(zhí)行任務(wù)。
執(zhí)行完成后,你會(huì)在你模塊的build->outputs->apk->tinkerPatch->release下看到你需要的補(bǔ)丁包patch_signed_7zip.apk。
- 最后調(diào)用
TinkerInstaller.onReceiveUpgradePatch(getApplicationContext(), path);, path傳入你補(bǔ)丁包的所在的路徑即可完成熱更新。
需要注意的是,執(zhí)行熱更新后,需要重啟程序才能生效!