如果看过我的前两篇博客Android插件化(二):OpenAtlas插件安装过程分析和Android插件化(三):OpenAtlas的插件重建以及使用时安装,就知道在插件的安装过程中OpenAtlas做了哪些事,那么插件的卸载就只需要把持久化和内存中的内容移除即可。
1.插件的卸载
插件卸载的对外接口为Atlas的uninstallBundle()方法:
|
|
- 先根据包名来获取插件对象(BundleImpl对象),如果不为空,则获取该插件对象源文件并删除,其中bundleImpl.getArchive().getArchiveFile()的值类似/data/data/cn.edu.zafu.atlasdemo/lib/libcom_lizhangqu_test.so这样的文件(如果可写的话).
- 之后调用BundleArchive的purge()进行清理工作,代码如下:
|
|
代码比较简单,先是遍历删除该插件各个版本(当前版本除外)的目录(类似”/data/data/cn.edu.zafu.atlasdemo/files/storage/com.lizhangqu.test/version.1”),之后清除revisions中的所有元素,再将当前版本的插件信息(currentRevision)插入到revisions中.
之后调用bundle.uninstall(),该方法的代码如下:
|
|
这个方法主要做了以下事情:
- 如果当前处于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()方法:
|
|
显然,这里只是先判断插件是否存在,如果存在的话,则调用BundleImpl的update()方法:
|
|
既然是更新,那就表示是新版本,所以调用BundleArchve的newRevision()方法:
|
|
显然,这里先是获取当前最新版本的版本号,然后在次基础上加1,之后新建一个基于当前插件文件archiveFile的BundleArchiveRevision对象,最后把这个对象放到revisions集合中,这样在下次查找插件版本的时候,找到的最新版本就是这里创建的插件版本。
不过我觉得逻辑上不太好的一点就是在新建了插件版本对象后,并没有立即切换,结合插件对象的重建过程可知,这意味着版本更新需要在下次起作用。显然,后面的一个改进方向是做到热更新。