Droid Plugin360 黑科技
DroidPlugin 是 360 手机助手在 Android 系统上实现的一种新的插件机制:它可以在无需安装、修改的情况下运行 APK 文件,此机制对改进大型 APP 的架构,实现多团队协作开发具有一定的好处。
定义:
HOST 程序:插件的宿主。
插件:免安装运行的 APK
限制和缺陷:
-
无法在插件中发送具有自定义资源的
Notification
,例如: a. 带自定义RemoteLayout的Notification b. 图标通过R.drawable.XXX指定的通知(插件系统会自动将其转化为Bitmap) -
无法在插件中注册一些具有特殊Intent Filter的
Service
、Activity
、BroadcastReceiver
、ContentProvider
等组件以供Android系统、已经安装的其他APP调用。 -
对
Activity
的LaunchMode支持不够好,Activity Stack管理存在一定缺陷。Activity的onNewIntent
函数可能不会被触发。 (此为BUG,未来会修复) -
缺乏对Native层的Hook,对某些带native代码的apk支持不好,可能无法运行。比如一部分游戏无法当作插件运行。
特点:
-
支持Androd 2.3以上系统
-
插件APK完全不需做任何修改,可以独立安装运行、也可以做插件运行。要以插件模式运行某个APK,你无需重新编译、无需知道其源码。
-
插件的四大组件完全不需要在Host程序中注册,支持Service、Activity、BroadcastReceiver、ContentProvider四大组件
-
插件之间、Host程序与插件之间会互相认为对方已经"安装"在系统上了。
-
API低侵入性:极少的API。HOST程序只是需要一行代码即可集成Droid Plugin
-
超强隔离:插件之间、插件与Host之间完全的代码级别的隔离:不能互相调用对方的代码。通讯只能使用Android系统级别的通讯方法。
-
支持所有系统API
-
资源完全隔离:插件之间、与Host之间实现了资源完全隔离,不会出现资源窜用的情况。
-
实现了进程管理,插件的空进程会被及时回收,占用内存低。
-
插件的静态广播会被当作动态处理,如果插件没有运行(即没有插件进程运行),其静态广播也永远不回被触发。
使用方法:
集成
在host中集成Droid Plugin项目非常简单:
-
我们只是需要将Droid Plugin当作一个lib工程应用到主项目中,然后:
-
在
AndroidManifest.xml
中使用插件的com.morgoo.droidplugin.PluginApplication
:<application android:name="com.morgoo.droidplugin.PluginApplication" android:label="@string/app_name" android:icon="@drawable/ic_launcher"
-
如果你使用自定义的
Application
,那么你需要在自定义的Application classonCreate
和attachBaseContext
方法中添加如下代码:@Override public void onCreate() { super.onCreate(); //这里必须在super.onCreate方法之后,顺序不能变 PluginHelper.getInstance().applicationOnCreate(getBaseContext()); } @Override protected void attachBaseContext(Context base) { PluginHelper.getInstance().applicationAttachBaseContext(base); }
-
将插件中
Libraries\DroidPlugin\AndroidManifest.xml
中所有的provider
对应的authorities
修改成自己的,默认为com.morgoo.droidplugin_stub_P00
,如下:<provider android:name="com.morgoo.droidplugin.stub.ContentProviderStub$StubP00" android:authorities="com.morgoo.droidplugin_stub_P00" android:exported="false" android:label="@string/stub_name_povider" />
可以修改为自己的包名,如:
com.example.droidplugin_stub_P00
防止跟其它本插件使用者冲突:<provider android:name="com.morgoo.droidplugin.stub.ContentProviderStub$StubP00" android:authorities="com.example.droidplugin_stub_P00" android:exported="false" android:label="@string/stub_name_povider" />
-
集成完成。
安装、卸载插件:
-
安装、更新插件,使用如下方法:
int PluginManager.getInstance().installPackage(String filepath, int flags)
说明:安装插件到插件系统中,
filepath
为插件apk路径,flags
可以设置为0,如果要更新插件,则设置为PackageManagerCompat.INSTALL_REPLACE_EXISTING
返回值及其含义请参见PackageManagerCompat
类中的相关字段。 -
卸载插件,使用如下方法:
int PluginManager.getInstance().deletePackage(String packageName,int flags);
说明:从插件系统中卸载某个插件,
packageName
传插件包名即可,flags
传0。 -
启动插件:启动插件的
Activity
、Service
等都和你启动一个以及安装在系统中的app一样,使用系统提供的相关API即可。组件间通讯也是如此。
实现原理:
请参见源码
支持:
任何问题可以在项目中提交bug报告,也可以发送邮件以寻求帮助。