笔者从事智能家具行业的开发工做,也是从公司创业团队工做到如今,对于公司的项目从1.0版本开始接手一直到如今,虽然说项目不是很大但麻雀虽小五脏俱全,在项目和团队的不断扩大、暴露出的问题也不段增多,组件化势在必行,本文就根据整个项目的发展,总结下组件化的实践流程;java
在进行组件化操做以前,先区分两个概念:模块化和组件化android
由上面的介绍知道,组件化针对更细更单一的业务,功能模块粒度较大,针对某个方面的总体业务,固然业务当中可能使用不少的独立组件,按照组件化的需求项目的架构进入3.0 网络
由上面的3.0版本架构知道,项目中包含多个功能组件和业务模块,在开发中要保证组件间不能耦合,业务木块依赖于组件,但业务模块之间也不能相互引用,不然违背了组件化的原则;架构
在咱们实际开发中app 构建形式为application,最终编译成APK文件,其他所依赖的Module编译形式为library,最终已arr形式寻在提供API调用,换句话说只要修改组件的编译形式便可实现单独编译的功能,因此在组件下建立gradle.properties文件用于控制构建形式app
isRunAlone = false
复制代码
在build.gradle中根据isRunAlone的变量修改构建形式框架
if (isRunAlone.toBoolean()) {
apply plugin: 'com.android.application'
} else {
apply plugin: 'com.android.library'
}
复制代码
if (isRunAlone.toBoolean()) {
applicationId "com.alex.kotlin.content"
}
复制代码
在组件化单独编译和总体编译时,注册清单中所须要的内容不一样,如单独编译须要额外的启动页,且单独编译时也休要配置不一样的Application,此时在main文件加下建立manifest/AndroidMenifest.xml文件,根据单独编译的须要设置内容。ide
sourceSets {
main {
if (isRunAlone.toBoolean()) {
manifest.srcFile 'src/main/manifest/AndroidManifest.xml'
} else {
manifest.srcFile 'src/main/AndroidManifest.xml'
}
}
}
复制代码
到此编译配置完成,在须要单独编译时只须要修改isRunAlone为true便可;模块化
由上面配置的两个注册清单文件中可见,在App总体编译时组件使用的是全局的Application,在单独编译时使用的是AutoApplication,你们都知道一个程序中只有一个Application类,那组件中须要初始化的代码都配置在本身的AutoApplication中,那总体编译时如何初始化呢?可能有同窗说总体编译时个组件和模块是可见的,直接调用AutoApplication类完成初始化,但此种状况主项目就没法实现模块的自由增减,并且当代码隔离时AutoApplication就不可见了,这里采用一种配置+反射的方式温馨化各组件的Application,具体实现以下:工具
abstract class BaseApp : Application(){
/**
* 初始化Module中的Application
*/
abstract fun initModuleApp(application: Application)
}
复制代码
class AutoApplication : BaseApp() {
override fun onCreate() { //单独编译时初始化
super.onCreate()
MultiDex.install(this)
AppUtils.setContext(this)
initModuleApp(this)
ServiceFactory.getServiceFactory().loginToService = AutoLoginService()
}
override fun initModuleApp(application: Application) { //总体编译
ServiceFactory.getServiceFactory().serviceIntelligence = AutoIntelligenceService()
}
}
复制代码
object AppConfig {
private const val BASE_APPLICATION = "com.pomelos.base.BaseApplication"
private const val CONTENT_APPLICATION = "com.alex.kotlin.content.ContentApplication"
private const val AUTO_APPLICATION = "com.alex.kotlin.intelligence.AutoApplication"
val APPLICATION_ARRAY = arrayListOf(BASE_APPLICATION, CONTENT_APPLICATION, AUTO_APPLICATION)
}
复制代码
public class GlobalApplication extends BaseApp {
@SuppressLint("StaticFieldLeak")
private static GlobalApplication instance;
public GlobalApplication() {}
@SuppressWarnings("AlibabaAvoidManuallyCreateThread")
@Override
public void onCreate() {
super.onCreate();
MultiDex.install(this);
AppUtils.setContext(this);
if (BuildConfig.DEBUG) {
//开启Debug
ARouter.openDebug();
//开启打印日志
ARouter.openLog();
}
//初始化ARouter
ARouter.init(this);
ServiceFactory.Companion.getServiceFactory().setLoginToService(new AppLoginService());
//初始化组件的Application
initModuleApp(this);
}
@Override
public void initModuleApp(@NotNull Application application) {
for (String applicationName : AppConfig.INSTANCE.getAPPLICATION_ARRAY()) { //遍历全部配置的Application
try {
Class clazz = Class.forName(applicationName); //反射执行
BaseApp baseApp = (BaseApp) clazz.newInstance(); //建立实例
baseApp.initModuleApp(application); // 执行初始化
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
}
}
复制代码
以上经过在AppConfig中配置全部的Application的路径,在主Application执行时反射建立每一个实例,调用对应的initModuleApp()完成全部的配置,不知有没有注意到在AutoApplication中一样在onCreate()中初始化了内容,此处是为了在单独编译时调用;源码分析
在项目中由于有时须要打包不一样需求的APK,因此我将login单独分离出成组件同一登陆行为,那么在特务模块依赖Login以后便可实现登陆功能,但每一个单独的业务独立编译时会产生多个APK,这些APK都须要获取登陆状态及跳转相应的首界面,那么在保证程序解耦的状况下如何实现呢?答案及时使用注册接口实现;
interface LoginToService {
/** * 实现登陆后的去向 */
fun goToSuccess()
}
复制代码
class ServiceFactory private constructor() {
companion object {
fun getServiceFactory(): ServiceFactory {
return Inner.serviceFactory
}
}
private object Inner {
val serviceFactory = ServiceFactory()
}
}
复制代码
var loginToService: LoginToService? = null
get() {
if (field == null) {
field = EmptyLoginService()
}
return field
}
复制代码
class AppLoginService : LoginToService { //App模块
override fun goToSuccess() {
val intent = Intent(AppUtils.getContext(), MainActivity::class.java)
intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK
AppUtils.getContext().startActivity(intent)
}
}
class AutoLoginService : LoginToService { // 智能模块
override fun goToSuccess() {
val intent = Intent(AppUtils.getContext(), AutoMainActivity::class.java)
intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK
AppUtils.getContext().startActivity(intent)
}
}
复制代码
ServiceFactory.getServiceFactory().serviceIntelligence = AutoIntelligenceService()
复制代码
override fun loadSuccess(loginBean: LoginEntity) {
ServiceFactory.getServiceFactory().loginToService?.goToSuccess()
}
复制代码
各组件经过向base组件中的ServiceFactory注册的方式,对外提供执行的功能,由于ServiceFactory单例调用,因此在其余组件中经过ServiceFactory获取注册的实例后便可执行方法,为了在减去组件或模块时防止报错,在base中一样提供了服务的空实现;
关于页面跳转推荐使用阿里的ARoute框架,详情见另外一篇文章:Android框架源码分析——以Arouter为例谈谈学习开源框架的最佳姿式
在通常项目中,主app的首界面都来自不一样的业务模块组成,最多见的就是使用不一样组件的Fragment和ViewPager组合,但此时主App须要获取组件中的Fragment实例,按照组件化的思想不能直接使用,不然主APP和组件、模块间又会耦合在一块儿,此处也是采用接口模式处理,过程和数据交互大体相同;
interface ContentService {
/** * 返回实例化的Fragment */
fun newInstanceFragment(): BaseCompatFragment?
}
// 内容模块实现
class ContentServiceImpl : ContentService {
override fun newInstanceFragment(): BaseCompatFragment? {
return ContentBaseFragment.newInstance() //提供Fragment对象
}
}
复制代码
ServiceFactory.getServiceFactory().serviceContent = ContentServiceImpl()
复制代码
mFragments[SECOND] = ServiceFactory.getServiceFactory().serviceContent?.newInstanceFragment()
复制代码
虽然经历组件化将代码解耦,但在开发中若是依赖的组件或模块中的方法老是可见,万一在开发中使用了其中的代码,那程序程序又会耦合在一块儿,如何能让组件和模块中的方法不可见呢?答案就在runtimeOnly依赖,他能够在开发过程当中隔离代码,在编译时代码可见
runtimeOnly project(':content') runtimeOnly project(':intelligence') 复制代码
runtimeOnly依赖实现了代码隔离,但对资源并无效果,使用中仍是可能会直接引用资源,为了防止这种现象,为每一个组件的资源加上特有的前缀
resourcePrefix "auto_"
复制代码
此时该Module下的资源都必须以auto_开头不然会警告;
因为项目中使用到了ContentProvider,(不了解的点击Android进阶知识树——ContentProvider使用和工做过程详解)在总体编译安装在手机后能够正常运行,此时要单独编译时老是提示安装失败,最终缘由就是两个Apk中的ContentProvider和权限一致致使,那如何保证单独编译和总体编译时权限不一样,从而安装成功呢?咱们首先在上面的连个Menifest文件中配置Provider
<provider
android:name=".database.MyContentProvider"
android:authorities="com.alex.kotlin.intelligence.database.MyContentProvider"
android:exported="false" />
复制代码
<provider
android:name=".database.MyContentProvider"
android:authorities="com.findtech.threePomelos.database.MyContentProvider"
android:exported="false" />
复制代码
这样两个权限不一样的Provider便可安装成功,在使用时须要根据权限执行ContentProvider,那么如何在代码中根据不一样编译类型,拼接对应的执行权限呢?此处使用在build.gradle中配置BuildConfig来处理,将权限直接配置在BuildConfig中,在使用时直接获取便可
if (isRunAlone.toBoolean()) {
buildConfigField 'String','AUTHORITY','"com.alex.kotlin.intelligence.database.MyContentProvider"'
}else {
buildConfigField 'String','AUTHORITY','"com.findtech.threePomelos.database.MyContentProvider"'
}
const val AUTHORITY = BuildConfig.AUTHORITY //使用
复制代码
解决上面的全部问题后,项目的组件化基本能够实现,但具体的划分粒度和细节,须要自身结合业务和经验去处理,可能有些须要直接分离组件,也可能小的功能须要放在base组件中共享,并且每一个人针对每一个项目的处理方式也不一样,只要理解组件化的思想和方式实现最终的需求便可;