Android插件化(4):OpenAtlasの插件的卸载与更新

核心提示:如果看过我的前两篇博客Android插件化(2):OpenAtlas插件安装过程分析和Android插件化(3):OpenAtlas的插件重建以及使用时安装,就知道在插件的安装过程中OpenAtlas做了哪些事,那么插件的卸载就只需要把持久化和内存中的内容移除即可。1.插件的卸载插件卸载的

如果看过我的前两篇博客 Android插件化(2):OpenAtlas插件安装过程分析 和 Android插件化(3):OpenAtlas的插件重建以及使用时安装 ,就知道在插件的安装过程中OpenAtlas做了哪些事,那么插件的卸载就只需要把持久化和内存中的内容移除即可。

1.插件的卸载

插件卸载的对外接口为Atlas的uninstallBundle()方法:

     public void uninstallBundle(String pkgName) throws BundleException {
Bundle bundle = Framework.getBundle(pkgName);
if (bundle != null) {
BundleImpl bundleImpl = (BundleImpl) bundle;
try {
File archiveFile = bundleImpl.getArchive().getArchiveFile();
if (archiveFile.canWrite()) {
archiveFile.delete();
}
bundleImpl.getArchive().purge();
File revisionDir = bundleImpl.getArchive().getCurrentRevision()
.getRevisionDir();
bundle.uninstall();
if (revisionDir != null) {
Framework.deleteDirectory(revisionDir);
return;
}
return;
} catch (Exception e) {
log.error("uninstall bundle error: " + pkgName + e.getMessage());
return;
}
}
throw new BundleException("Could not uninstall bundle " + pkgName + ", because could not find it");
}
  • 先根据包名来获取插件对象(BundleImpl对象),如果不为空,则获取该插件对象源文件并删除,其中bundleImpl.getArchive().getArchiveFile()的值类似/data/data/cn.edu.zafu.atlasdemo/lib/libcom_lizhangqu_test.so这样的文件(如果可写的话).
  • 之后调用BundleArchive的purge()进行清理工作,代码如下:
     @Override
public void purge() throws Exception {
if (this.revisions.size() > 1) {
long revisionNum = this.currentRevision.getRevisionNum();
for (Long longValue : this.revisions.keySet()) {
long longValue2 = longValue.longValue();
if (longValue2 != revisionNum) {
File file = new File(this.bundleDir, "version."
+ String.valueOf(longValue2));
if (file.exists()) {
Framework.deleteDirectory(file);
}
}
}
this.revisions.clear();
this.revisions.put(Long.valueOf(revisionNum), this.currentRevision);
}
}

代码比较简单,先是遍历删除该插件各个版本(当前版本除外)的目录(类似"/data/data/cn.edu.zafu.atlasdemo/files/storage/com.lizhangqu.test/version.1"),之后清除revisions中的所有元素,再将当前版本的插件信息(currentRevision)插入到revisions中.

之后调用bundle.uninstall(),该方法的代码如下:

     @Override
public synchronized void uninstall() throws BundleException {
if (this.state == BundleEvent.INSTALLED) {
throw new IllegalStateException("Bundle " + toString() + " is already uninstalled.");
}
if (this.state == BundleEvent.RESOLVED) {
try {
stopBundle();
} catch (Throwable th) {
Framework.notifyFrameworkListeners(BundleEvent.STARTED, this, th);
}
}
//为什么this.state=BundleEvent.INSTALLED而不是this.state=BundleEvent.UNINSTALLED?
this.state = BundleEvent.INSTALLED;
new File(this.bundleDir, "meta").delete();
if (this.classloader.originalExporter != null) {
this.classloader.originalExporter.cleanup(true);
this.classloader.originalExporter = null;
}
this.classloader.cleanup(true);
this.classloader = null;
Framework.bundles.remove(this);
Framework.notifyBundleListeners(BundleEvent.UNINSTALLED, this);
this.context.isValid = false;
this.context.bundle = null; }

这个方法主要做了以下事情:

+ 如果当前处于BundleEvent.RESOLVED状态,则需要调用stopBundle()使当前的状态置为BundleEvent.STOPPED并通知监听者; + 之后将状态置为BundleEvent.INSTALLED,然后删除类似/data/data/cn.edu.zafu.atlasdemo/files/storage/com.lizhangqu.test/meta这样的插件元数据文件; + 清除BundleClassLoader中的originalExporter,这个表示BundleClassLoader中对外输出的BundleClassLoader对象; + 调用BundleClassLoader的cleanup()方法以清除包的依赖关系,但是实际上OpenAtlas并未实现包的依赖这个功能,所以到了ACDD这个方法中就只是将BundleImpl在BundleClassLoader中的引用去掉了; + 之后将BundleClassLoader对象置为null,然后将当前插件重Framework.bundles中移除 + 通知监听者,当前插件已经卸载

再回到BundleArchive的purge()方法中,会将插件版本目录删除,这样就话把这个插件版本下的所有数据都清除,包括安装插件时新建的插件版本元数据文件.

到这里为止,插件的卸载就完成了,整个过程非常简单。

2.插件的更新

插件的更新接口为Atlas.updateBundle()方法:

public void updateBundle(String pkgName, File mBundleFile) throws BundleException {
if (!mBundleFile.exists()) {
throw new BundleException("file not found" + mBundleFile.getAbsolutePath());
}
Bundle bundle = Framework.getBundle(pkgName);
if (bundle != null) {
bundle.update(mBundleFile);
return;
}
throw new BundleException("Could not update bundle " + pkgName
+ ", because could not find it");
}

显然,这里只是先判断插件是否存在,如果存在的话,则调用BundleImpl的update()方法:

   @Override
public synchronized void update(File bundleFile) throws BundleException {
//这里应该是写错了,应该修改成BundleEvent.UNINSTALLED或者this.state!=BundleEvent.UNINSTALLED
//额,确实没有错,因为在uninstall之后有this.state=BundleEvent.INSTALLED;只是这种逻辑很奇怪
if (this.state == BundleEvent.INSTALLED) {
throw new IllegalStateException("Cannot update uninstalled bundle "
+ toString());
}
try {
this.archive.newRevision(this.location, this.bundleDir, bundleFile);
} catch (Throwable e) {
throw new BundleException("Could not update bundle " + toString(),
e);
}
}

既然是更新,那就表示是新版本,所以调用BundleArchve的newRevision()方法:

   /**
*
* @param packageName
* @param bundleDir
* @param archiveFile 类似"/data/data/cn.edu.zafu.altasdemo/lib/libcom_lizhangqu_test.so"这样的文件
* @return
* @throws IOException
*/
@Override
public BundleArchiveRevision newRevision(String packageName, File bundleDir, File archiveFile)
throws IOException {
long revision = 1 + this.revisions.lastKey().longValue();
BundleArchiveRevision bundleArchiveRevision = new BundleArchiveRevision(
packageName, revision, new File(bundleDir, "version."
+ String.valueOf(revision)), archiveFile);
this.revisions.put(Long.valueOf(revision), bundleArchiveRevision);
return bundleArchiveRevision;
}

显然,这里先是获取当前最新版本的版本号,然后在次基础上加1,之后新建一个基于当前插件文件archiveFile的BundleArchiveRevision对象,最后把这个对象放到revisions集合中,这样在下次查找插件版本的时候,找到的最新版本就是这里创建的插件版本。

不过我觉得逻辑上不太好的一点就是在新建了插件版本对象后,并没有立即切换,结合插件对象的重建过程可知,这意味着版本更新需要在下次起作用。显然,后面的一个改进方向是做到热更新。

Android插件化(4):OpenAtlasの插件的卸载与更新的更多相关文章

  1. Android插件化(二):OpenAtlas插件安装过程分析

    Android插件化(二):OpenAtlas插件安装过程分析   转 https://www.300168.com/yidong/show-2788.html   核心提示:在前一篇博客 Andro ...

  2. Android插件化(三):OpenAtlas的插件重建以及使用时安装

    Android插件化(三):OpenAtlas的插件重建以及使用时安装 转 https://www.300168.com/yidong/show-2778.html    核心提示:在上一篇博客 An ...

  3. android开发环境之ADT安装,卸载,更新

    http://hj198703.iteye.com/blog/1316991   1.官网:http://developer.android.com/sdk/index.html2.ADT组件在线安装 ...

  4. Android组件化和插件化开发

    http://www.cnblogs.com/android-blogs/p/5703355.html 什么是组件化和插件化? 组件化开发就是将一个app分成多个模块,每个模块都是一个组件(Modul ...

  5. android插件化之路

    概论  插件式开发通俗的讲就是把一个很大的app分成n多个比较小的app,其中有一个app是主app.基本上可以理解为让一个apk不安装也可以被运行.只不过这个运行是有很多限制的运行,所以才叫插件. ...

  6. Android插件化

    http://www.androidblog.cn/index.php/Index/detail/id/16# Android Hotfix 新方案——Amigo 源码解读 https://www.d ...

  7. android插件化简述

    2015年是Android插件化技术突飞猛进的一年,随着业务的发展各大厂商都碰到了Android Native平台的瓶颈: 从技术上讲,业务逻辑的复杂导致代码量急剧膨胀,各大厂商陆续出到65535方法 ...

  8. Android插件化框架

    1.   dynamic-load-apk/DL动态加载框架 是基于代理的方式实现插件框架,对 App 的表层做了处理,通过在 Manifest 中注册代理组件,当启动插件组件时,首先启动一个代理组件 ...

  9. Android 插件化开发(四):插件化实现方案

    在经过上面铺垫后,我们可以尝试整体实现一下插件化了.这里我们先介绍一下最简单的实现插件化的方案. 一.最简单的插件化实现方案 最简单的插件化实现方案,对四大组件都是适用的,技术面涉及如下: 1). 合 ...

随机推荐

  1. java - day019 - 反射

    网络程序,难点在线程 反射 reflect 实用 类对象 来执行反射操作 反射获得一个类的定义信息 反射创建对象 反射调用成员变量, 方法 方法 获得类对象的三种方式 A.class Class.fo ...

  2. CISCO设备配置SSH 登陆

    1. 设置设备域名CISCO ip domain name CISCO2.创建密钥Server(config)#crypto key generate rsa The name for the key ...

  3. Sql Server 2017 安装问题记录

    记录了我在虚拟机中安装Sql server 2017遇到的一些问题. 安装环境: Sql server 2017 + Windows Server 2012 R2 提供两个网上的下载链接: https ...

  4. 用Jmeter做性能测试,之后报表展示

    https://octoperf.com/blog/2017/10/19/how-to-analyze-jmeter-results/ 看到性能测试平台的开发,我在想需要什么功能,报表需要什么样子的 ...

  5. JavaScript(ES6之前)数组方法总结

    一.数组的创建 1.使用 Array 构造函数 var arr1 = new Array(); // 创建一个空数组 var arr2 = new Array(20); // 创建一个包含20项的数组 ...

  6. 4.kafka生产者---向Kafka中写入数据(转)

    转:  https://www.cnblogs.com/sodawoods-blogs/p/8969513.html (1)生产者概览 (1)不同的应用场景对消息有不同的需求,即是否允许消息丢失.重复 ...

  7. 个性化召回算法实践(五)——item2vec

    item2vec将用户的行为序列转化成item组成的句子,模仿word2vec训练word embedding将item embedding.基本思想是把原来高维稀疏的表示方式(one_hot)映射到 ...

  8. jQuery基础知识1

    jquery的概念 js query jquery库 封装了大量js,封装js的入口函数.兼容性问题.DOM操作.事件.ajax 使用jquery 下载包 引用 <script src=&quo ...

  9. Docker那些事儿之编排工具docker-compose

    前面已经讲解过docker的一些基础使用,镜像创建的操作过程,如果大量容器需要同时部署,一个一个容器进行服务器上的部署,估计要疯掉,在使用上我们需要找到更好更便捷的使用方式,今天要讲解的容器编排工具d ...

  10. linux 安装MySql 5.7.20 操作步骤【亲测】

    一. #卸载系统自带的Mariadb[root@master ~]# rpm -qa|grep mariadbmariadb-libs-5.5.44-2.el7.centos.x86_64[root@ ...