部分热修复框架的对比如下表所示
3、热修复实践
最终我们App的热修复方案选择的是AndFix,原因有三: (1)AndFix支持android2.3-6.0,所以在机型上的是适配上是没问题的; (2)AndFix是由阿里开源的,并且持续维护中,目前不少公司已经使用其作为自身App的热修复方案; (3)通过修改Dex加载顺序的方式实现热修复需要重新启动App,并且相应的开源框架多多少少存在着问题,没有持续的维护;
因此我们最终选择了AndFix作为我们的开源方案。具体的AndFix集成方式可参考github中AndFix的介绍
这里简单介绍一下具体的继承流程 (1)在App的Application的onCreate方法中执行AndFix的初始化操作; (2)判断服务器端是否有可更新的热修复差异包 (3)若无则直接退出,若有则下载并执行修复动作 (4)修复完成之后删除下载的补丁差异包 (5)在判断服务器端是否有可更新的补丁包的时候可添加灰度,如版本,渠道,用户等,实现对补丁包定制化的修复
另外需要说明的是:若一个版本中存在着多个bug,则一般的都是让后一个补丁包覆盖前一个补丁包,并删除前一个补丁包,简单来说就是对于每一个版本至多有一个补丁包。
最后贴上App端AndFix的实现源码:
/**
- Created by aaron on 2016/3/7.
- 主要用于实现热修复逻辑
- 采用阿里巴巴开源框架-andfix
*/
public class AndfixManager {
public static final String TAG = AndfixManager.class.getSimpleName();
// AndfixManager单例对象
private static AndfixManager instance = null;
// 补丁文件名称
public static final String PATCH_FILENAME = “/patchname.apatch”;
public static PatchManager patchManager = null;
private AndfixManager() {}
/**
- 线程安全之懒汉模式实现单例模型
- @return
*/
public static synchronized AndfixManager getInstance() {
return instance == null ? new AndfixManager() : instance;
}
/**
- 执行andfix初始化操作
*/
public static void init(Context mContext) {
if (mContext == null) {
L.i(“初始化热修复框架,参数错误!!!”);
return;
}
patchManager = new PatchManager(mContext);
// 初始化patch版本,这里初始化的是当前的App版本;
patchManager.init(VersionUtils.getVersionName(mContext));
// 加载已经添加到PatchManager中的patch
patchManager.loadPatch();
downLoadAndAndPath(mContext);
}
/**
- 请求服务器获取补丁文件并加载
*/
public static void downLoadAndAndPath(final Context mContext) {
// 请求服务器获取差异包
ExtInterface.GetShContent.Request.Builder request = ExtInterface.GetShContent.Request.newBuilder();
// 获取本地保存的补丁包版本号
final String patchVersion = AndfixSp.getPatchVersion(mContext);
L.i(TAG, “patchVersion:” + patchVersion);
if (!TextUtils.isEmpty(patchVersion)) {
request.setShVersion(patchVersion);
} else {
request.setShVersion(“0”);
}
NetworkTask task = new NetworkTask(Cmd.CmdCode.GetShContent_SSL_VALUE);
task.setBusiData(request.build().toByteArray());
NetworkUtils.executeNetwork(task, new HttpResponse.NetWorkResponse() {
@Override
public void onSuccessResponse(UUResponseData responseData) {
if (responseData.getRet() == 0) {
try {
ExtInterface.GetShContent.Response response = ExtInterface.GetShContent.Response.parseFrom(responseData.getBusiData());
// 若返回成功,则更新脚本下载补丁包
if (response.getRet() == 0) {
ByteString zipDatas = response.getContent();
// 数据解压缩
byte[] oriDatas = GZipUtils.decompress(zipDatas.toByteArray());
String patchFileName = mContext.getCacheDir() + PATCH_FILENAME;
L.i(TAG, “patchFileName:” + response.getShVersion());
// 将byte数组数据写入文件
boolean boolResult = getFileFromBytes(patchFileName, oriDatas);
// 写入文件成功则加载
if (boolResult) {
patchManager.removeAllPatch();
patchManager.addPatch(patchFileName);
// 保存补丁版本号
AndfixSp.putPatchVersion(mContext, response.getShVersion());
// 删除补丁文件
File files = new File(patchFileName);
if (files.exists()) {
files.delete();
}
}
} else {
// -1 请求失败
// 1 请求成功,但是没有更新版本的脚本
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
@Override
public void onError(VolleyError errorResponse) {
}
@Override
public void networkFinish() {
}
});
}
/**
- 根据数组获取文件
- @param path
- @param oriDatas
*/
public static boolean getFileFromBytes(String path, byte[] oriDatas) {
boolean result = false;
最后
我的面试经验分享可能不会去罗列太多的具体题目,因为我依然认为面试经验中最宝贵的不是那一个个具体的题目或者具体的答案,而是结束面试时,那一刻你的感受以及多天之后你的回味~
很多人在刚接触这个行业的时候或者是在遇到瓶颈期的时候,总会遇到一些问题,比如学了一段时间感觉没有方向感,不知道该从那里入手去学习,对此我整理了一些资料,需要的可以免费分享给大家
在这里小编分享一份自己收录整理上述技术体系图相关的几十套腾讯、头条、阿里、美团等公司的面试题,把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分。
【Android核心高级技术PDF文档,BAT大厂面试真题解析】
【算法合集】
【延伸Android必备知识点】
【Android部分高级架构视频学习资源】
**Android精讲视频领取学习后更加是如虎添翼!**进军BATJ大厂等(备战)!现在都说互联网寒冬,其实无非就是你上错了车,且穿的少(技能),要是你上对车,自身技术能力够强,公司换掉的代价大,怎么可能会被裁掉,都是淘汰末端的业务Curd而已!现如今市场上初级程序员泛滥,这套教程针对Android开发工程师1-6年的人员、正处于瓶颈期,想要年后突破自己涨薪的,进阶Android中高级、架构师对你更是如鱼得水,赶快领取吧!
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!
id开发工程师1-6年的人员、正处于瓶颈期,想要年后突破自己涨薪的,进阶Android中高级、架构师对你更是如鱼得水,赶快领取吧!
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!