Flutter里的Dart代码和资源文件都可修复,请老铁们在接入前,先按照tinker或sophix的文档配置好,并调试确认可以正常修复原生代码,再接入hotfixflutter的配置。如果直接接入,出了问题,你会迷茫到底是tinker或sophix的问题还是hotfixflutter的问题。
混栈开发之Android端Flutter热更新-Sophix篇(二)
Google从性能和安全两方面考虑,暂时不会推出热更新,把这个功能交给用户和第三方去处理。结合公司不愿提供任何资源,只能走捷径的方式使用Tinker去实现Flutter的热更新。
Native项目可以接入Tinker进行热更新,而且有Bugly做为补丁版本控制台,来上传下发补丁,统计数量,不需要再去实现,省了不少事。
接入Flutter模块,修改Dart代码后,执行buildTinkerPatchRelease,生成patch_signed_7zip.apk补丁包,解开patch_signed_7zip.apk,里面也生成了Flutter模块的补丁so包。测试直接使用Tinker进行热更新,Dart代码的修改并未生效。
由于Flutter有自己的一套so加载流程,Tinker无法加载到Flutter的补丁so包。分析下Flutter的so加载流程,在FlutterLoader类里可以通过反射字段aotSharedLibraryName把它set进去,这样就可以实现加载补丁so文件,测试Dart代码的修改生效。Flutter新版本增加了FlutterApplicationInfo,但是不影响反射的使用,反射FlutterApplicationInfo里的aotSharedLibraryName,把路径set进去,再塞给FlutterLoader(感谢@heroghost的提醒)。
但是在集成FlutterBoost后,这样就行不通了。因为Flutter的初始化封装在FlutterBoost类里,FlutterBoost里又new了一个FlutterEngine引擎类,传入一个空字符数组的FlutterShellArgs。需要我们把反射方法放在里面的初始化方法后执行
FlutterMain.startInitialization(mPlatform.getApplication());
这就需要利用插桩,找到初始化方法,把反射方法插到初始化方法后面。测试生效。
整个修复过程,都是利用Flutter自身加载so文件去实现,所以不会出现兼容性和安全性的问题,而且也不会对系统性能有任何大的损耗。同时,Tinker开源,可以方便的查阅Tinker的源码。
Flutter版本1.22.5,Dart版本2.10.4。Flutter低版本抓紧升级,不要用命令去升级Flutter,最好直接去官网下载zip解压,能省去好多问题。
Gradle版本5.4.1,Gradle Plugin版本3.4.1。
无需关心是否集成FlutterBoost
无需关心FlutterPatch类
无需在Flutter初始化后编写
FlutterPatch.flutterPatchInit(this);
新增Sophix支持Flutter热更新
-
根配置添加,repositories下 (把jcenter的配置清除掉)
maven { url 'https://jitpack.io' }
dependencies下
classpath 'com.github.magicbaby810:hannibal:1.0.7.3'
如果纯flutter项目换成这个,其他不变
classpath 'com.github.magicbaby810:hannibalx:0.1'
-
在app gradle里配置
apply plugin: 'hannibal'
dependencies下
implementation 'com.sk.flutterpatch:flutterpatch:0.0.7'
另外说明,纯flutter项目,自动生成的tinkerId会出现补丁和基准包的id不一致,所以请各位自行设置tinkerId, 注掉autoGenerateTinkerId 纯Flutter项目热更新demo
tinkerId = "1.0.0-base" // autoGenerateTinkerId = true
第一次运行请先按步骤走下
-
down下来后,先打开flutterhotfixmodule项目,open->HotFixFlutter->flutterhotfixmodule,别管报错信息,直接打开pubspec.yaml,点击Pub get,执行完成。
-
打开HotFixFlutter,切换到Project下,打开根目录的settings.gradle,把下面的配置copy进去。
注意一定要填对路径,这个是我demo的路径,如果你用自己的项目跑的话,就需要把你的路径给放进来,比如'/xxx/.android/include_flutter.groovy'Native项目和Flutter项目在同一个目录下,如下配置
setBinding(new Binding([gradle: this])) evaluate(new File(settingsDir.parentFile, '/HotFixFlutter/flutterhotfixmodule/.android/include_flutter.groovy')) include ':flutterhotfixmodule'
Native项目和Flutter项目不在同一个目录下,如下配置
setBinding(new Binding([gradle: this])) evaluate(new File(settingsDir.parentFile, flutterhotfixmodule/.android/include_flutter.groovy')) include ':flutterhotfixmodule' project(':flutterhotfixmodule').projectDir = new File('../flutterhotfixmodule')
点击Sync Now,执行完成,会看到项目结构变成田格样式
-
在app的gradle里,配置下面flutter、flutterboost,以及flutterpatch的依赖,再次Sync Now。
implementation project(':flutter') implementation 'com.sk.flutterpatch:flutterpatch:0.0.7'
如果要测试flutterpatch,在flutterpatch模块的gradle里配置如下
compileOnly project(':flutter')
如果是老手,已接过Tinker,无需再看下面步骤。新手接入,可以跟着我这个步骤走下,腾讯的官方文档乱七八糟的
-
把bugly id复制到bugly初始化里面
Bugly.init(this, "你的bugly id", true);
运行gradle下面的assembleRelease任务。如果有error,请先clean project再试。
执行完成,安装build->bakApk->带有日期文件夹->app-release.apk。
-
去flutterhotfixmodule项目下修改dart代码,以及添加加载图片资源。修改完后回到HotfixFlutter项目下,把build->bakApk下生成目录上的安装日期抄写到tinker-support.gradle里的baseApkDir里。执行
-
进入 bugly官网,打开热更新页面,点击发布新补丁,找到build->outputs->patch->patch_signed_7zip.apk,上传完成,点击全量设备(只限测试,别整个生产的bugly id进来啊),立即下发。稍微等待那么一小会,杀掉应用,再重新打开,会出现
代表补丁已经打上去了,杀掉应用,再次打开进去flutter页面,修复成功!
这个不需要细说了吧,阿里的接入文档,还是很清晰的,配置完就ok了。
- 比Tinker多个
SophixStubApplication
- 打补丁需要用到
SophixPatchTool
,一定要区分好新旧包 - 还可以利用
Sophix调试工具V3
app快速加载补丁 - 在demo里操作Sophix的时候,记得在
AppApplication
里注掉attachBaseContext
这个方法