Android AutoService 组件化

作者:芩qin

1.前言

随着 App 的业务增加、版本迭代以及冗余的 “远古时期” 代码,App 代码变得臃肿增量叠加、开发者需要了解各个功能、单测功能编译时长、没有统一快速开发框架,代码复用性低,组件化开发就很有必要。

2.组件化架构

1).组件化架构的思想

组件化开发框架可以细化为不同的部分,包括 Android UI、网络请求、数据库持久化、图片处理、View、工具类、sdk、内部统一风格组件等;框架包括但不限于通用功能,如果是部门内部项目中通用的功能,也可以独立出来成为一个通用的库存在。

组件化架构图

2).组件化有哪些方案、各自的优势

① ARouter:基因中自带支持从webview中调用、不用互相注册(不用知道需要调用的app的进程名称等信息)等;

② ComponentCaller: 集成简单、功能丰富、全程监控、改造老项目成本低等;

③ Google AutoService:Google推荐、继承简单、功能强大等;

3).AutoService 组件化实现

原理:AutoService会自动在META-INF文件夹下生成Processor配置信息文件,该文件里就是实现该服务接口的具体实现类。而当外部程序装配这个模块的时候, 就能通过该jar包META-INF/services/里的配置文件找到具体的实现类名,并装载实例化,完成模块的注入。

① 添加依赖

implementation 'com.google.code.gson:gson:2.8.6'

② 添加 Javapoet 常用 api

annotationProcessor 'com.google.code.gson:gson:2.8.6'
//Kotlin 需要kapt支持
apply plugin: 'kotlin-android'apply 
plugin: 'kotlin-android-extensions'
//kapt插件、会有很多问题、博客地址:https://www.jianshu.com/p/b58d733bc54eapply 
plugin: 'kotlin-kapt'

③ 使用 @AutoService 注解

//第一步 创建下沉接口
interface IWebViewService {    
fun startWebActivity(context: Context,title: String,url: String)
    fun startWebFragment(url: String): Fragment  
  fun starLocalTestHtml(context: Context)
}// 第二步 实现接口
@AutoService(IWebViewService::class)
class WebViewServiceImpl : IWebViewService { 
   override fun startWebActivity(context: Context,url: String) {  
      WebActivity.create(context,title,url)   
 }   
 override fun startWebFragment(url: String): Fragment {     
   return WebFragment.create(url) 
   }   
 override fun starLocalTestHtml(context: Context) {    
    WebActivity.createHtml(context)  
  }
}// 第三步 查找实例、进行通信
binding.starWebActivity.setOnClickListener {  
// AutoService工具类找实现
 AutoService.load(IWebViewService::class.java)?.apply {   
     starLocalTestHtml(this@AccountActivity)  
  }
}
object AutoService {  
  fun <S> load(clazz: Class<S>): S? {    
    val service = ServiceLoader.load(clazz).iterator()    
    try {      
      if (service.hasNext()) {  
              return service.next()     
       }       
 } catch (e: Exception) {     
       e.printStackTrace()    
    }     
   return null  
  }
}

以上就完成了 组件化的初步构建,结构如下图

3. WebView 组件封装

1).WebView 的组成部分

2).创建视图

① 创建 WebActivity & WebFragment & BaseWebView & IWebCallBack
IWebCallBack:Web页面打开时 WebViewClient 和 WebChromeClient 事件监听。
WebActivity: Web页面的入口、IWebCallBack 实现监听并统一管理页面。
WebFragment:返回一个统一事件处理的 Fragment 页面。
BaseWebView:自定义 WebView 统一配置 WebSettings 属性、由 IWebCallBack 将 WebViewClient 和 WebChromeClient 事件回调给 WebActivity 或 WebFragment;

并配置 JavascriptInterface 方法用于接收 Web 事件、统一处理。

3).跨进程通信

Web 页面所需要的内存比较大,为了避免 WebView 的OOM造成 App 的崩溃,需将Web 页面运行在独立的进程,跨进程通信使用 AIDL。

① 为了方便管理,首先进行分包 MainProcess 和 WebProcess;Web页面是运行在 web 进程中,而响应 web 页面的事件及处理是在 main 进程中,进程切换借助 AIDL ,则创建一个

IWebProToMainPro 的 aidl 接口如下:

// Declare any non-default types here with 
import statementsimport com.hlc.common.IMainProToWebPro;
interface IWebProToMainPro {   
 /**   
  * Demonstrates some basic types that you can use as parameters   
  * and return values in AIDL.    
  */   
 void handleWebCommand(String commandName,String jsonParams,IMainProToWebPro callBack);
}

其位置在 common 层(可以在 web 模块中,但事件的调度需要在 app 中,此项目 app 为空壳)。

4).命令模式

为了统一管理 web 页面的事件,则使用命令模式:只定义一个 JavascriptInterfacefun 接口去响应 web 页面,服务端通过下发命令进行事件分发,BaseWebView 定义如下:

//接受 web 事件 
@JavascriptInterfacefun
 takeNativeAction(jsParams: String) { 
   Timber.tag(TAG).d(jsParams)  
  if (jsParams.isNotBlank()) {    
    val jsonParams = Gson().fromJson(jsParams,JsonParams::class.java) 
       Timber.tag(TAG).d(Gson().toJson(jsonParams.param))     
       WebViewCommandDispatcher.execute(jsonParams.name,jsonParams.param,object : IMainProToWebPro.Stub() {    
            override fun onResult(callBackName: String?,response: String?) {  
                  Timber.tag(TAG).d("callBackName:$callBackName,Response:$response")   
             }        
    })  
  }
}

同时也在 common 创建一个 Command 接口,由实现类去处理事件、响应web请求。

interface Command {  
  fun name(): String   
 fun execute(params: String,callBack:IMainProToWebPro?)
}

5).事件分发

首先在主进程中创建命令管理器并实现 aidl 接口服务桩 IWebProToMainPro.Stub 类,然后通过

ServiceLoader 去查找所有的 Command 实现类,根据服务器的命令进行事件分发:

/**
 * 主进程命令管理器
 * IWebProToMainPro.aidl 全称:IWebViewProcessToMainProcessInterface.aidl 
 * WebViewProcess 到 MainProcess 的接口(AIDL)
 * @author hlc 
 */
object MainProcessCommandManager : IWebProToMainPro.Stub() {  
  private const val TAG = "MainProCommandManager"  
  private val commands = mutableMapOf<String,Command>() 
   /**  
   * 查找所有的Command   
   */   
 init {   
     val serviceLoader = ServiceLoader.load(Command::class.java)     
     Timber.tag(TAG).d("serviceLoader hasNext :${serviceLoader.iterator().hasNext()}")   
     for (command in serviceLoader) {      
       if (!commands.contains(command.name())) {     
           commands[command.name()] = command        
    }     
   }  
  }  
  /**  
   * 解析执行命令 
   */    
override fun handleWebCommand(commandName: String?,jsonParams: String?,callBack: IMainProToWebPro?) {   
     Timber.tag(TAG).d(jsonParams)     
   if (!commandName.isNullOrBlank() && !jsonParams.isNullOrBlank()) {    
        commands[commandName]?.execute(jsonParams,callBack) 
       }  
  }
}

然后要使用AIDL,Service需要以 aidl 文件的方式提供服务接口,AIDL 工具将生成一个相应的 java 接口,并且在生成的服务接口中包含一个功能调用的 Stub 服务桩类,Service 的实现类需要去绑定这个stub服务桩类:

/** * 主进程命令Service、用于连接WebViewProcess和MainProcess
 * @author hlc 
 */
class MainProcessCommandService : Service() { 
   override fun onBind(intent: Intent?): IBinder? {  
      return MainProcessCommandManager  
  }
}

在 JavascriptInterface 中接收了 web 页面的事件必然需要分发到主进程或其他进程处理,此时在 web 进程中需要创建事件分发器监控 Service 的存活状态:

/** 
* WebView事件分发 
* @author hlc 
*/
object WebViewCommandDispatcher : ServiceConnection { 
   private var iWebProToMainPro: IWebProToMainPro? = null  
  fun initAidlConnection() {    
    BaseApplication.baseApplication?.also {    
        val intent = Intent(it,MainProcessCommandService::class.java) 
          //开启Service时要通过setAction来启动,因为Service在另一个程序,所以用显性的话会找不到,
          //只能通过隐性来启动 
          it.bindService(intent,this,Context.BIND_AUTO_CREATE)   
     }  
  }   
 override fun onServiceConnected(name: ComponentName?,service: IBinder?) {  
      iWebProToMainPro = IWebProToMainPro.Stub.asInterface(service)  
  }  
  override fun onServiceDisconnected(name: ComponentName?) { 
       iWebProToMainPro = null    
    //重新连接服务      
  initAidlConnection()  
  }  
  override fun onBindingDied(name: ComponentName?) {  
      iWebProToMainPro = null    
    //重新连接服务     
   initAidlConnection() 
   }  
  fun execute(commandName: String,params: String,callBack: IMainProToWebPro) {  
      iWebProToMainPro?.handleWebCommand(commandName,params,callBack)  
  }
}

此外需要在 web 页面创建时开启服务,由此 web 进程到主进程的通信就完成,比如打开登录页面 Command:

@AutoService(Command::class)
class LoginCommand : Command {  
  private var mCallBack: IMainProToWebPro? = null  
  private var mCallBackName: String = "" 
   init {    
    LiveEventBus.get(EventKey.LOGIN_RESULT,String::class.java).observeForever {    
        Timber.tag(TAG).d("登录结果:UserName->$it")   
         mCallBack?.onResult(mCallBackName,it)      
  }   
 }  
 override fun name(): String = COMMAND_NAME   
 override fun execute(params: String,callBack: IMainProToWebPro?) {   
   Timber.tag(TAG).d(params)     
   mCallBack = callBack   
   val param = Gson().fromJson(params,LoginParam::class.java) as LoginParam    
   BaseApplication.baseApplication?.also {      
    val intent = Intent()    
    intent.component = ComponentName(it,param.targetClass)    
    intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK        
    it.startActivity(intent)      
      } 
    }   
 companion object {   
   private const val TAG = "LoginCommand"     
   private const val COMMAND_NAME = "openLoginPage"
    }
}

6).返回结果

在登录成功后需要返回 web 页面,并更新页面信息,此时亦是跨进程通信,返回结果通常是使用callBack,创建 IMainProToWebPro 的 aidl 接口,并在 IWebProToMainPro 的handleWebCommand 方法中以参数形式传给 Command 实现,然后将其结果返回并更新页面。

总结

① AutoService 的源码、原理分析;
② 跨进程原理、AIDL 原理、Binder 机制;
③ WebView 的功能、接口使用;

最后推荐一份学习秘籍,有 Android 学习文档+Android 视频+Android 面试题+Android 核心笔记,希望可以帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,可以分享给身边好友一起学习,如果有需要可以私信回复 666即可!!!

原文地址:https://blog.csdn.net/weixin_61845324

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


更新Android SDK到3.0版本时,遇到Failed to rename directory E:\android\tools to E:\android\temp\ToolPackage.old01问题,导致无法更新,出现该问题的原因是由于3.0版本与较早的sdk版本之间文件结构有冲突,解决
Android 如何解决dialog弹出时无法捕捉Activity的back事件 在一些情况下,我们需要捕捉back键事件,然后在捕捉到的事件里写入我们需要进行的处理,通常可以采用下面三种办法捕捉到back事件: 1)重写onKeyDown或者onKeyUp方法 2)重写onBackPressed方
Android实现自定义带文字和图片的Button 在Android开发中经常会需要用到带文字和图片的button,下面来讲解一下常用的实现办法。一.用系统自带的Button实现 最简单的一种办法就是利用系统自带的Button来实现,这种方式代码量最小。在Button的属性中有一个是drawable
Android中的&quot;Unable to start activity ComponentInfo&quot;的错误 最近在做一款音乐播放器的时候,然后在调试的过程中发现一直报这个错误&quot;Unable to start activity ComponentInfo&quot;,从字面
Android 关于长按back键退出应用程序的实现最近在做一个Android上的应用,碰到一个问题就是如何实现长按back键退出应用程序。在网上查找了很多资料,发现几乎没有这样的实现,大部分在处理时是双击back键来退出应用程序。参考了一下双击back键退出应用程序的代码,网上主流的一种方法是下面
android自带的时间选择器只能精确到分,但是对于某些应用要求选择的时间精确到秒级,此时只有自定义去实现这样的时间选择器了。下面介绍一个可以精确到秒级的时间选择器。 先上效果图: 下面是工程目录: 这个控件我也是用的别人的,好像是一个老外写的,com.wheel中的WheelView是滑动控件的主
Android平台下利用zxing实现二维码开发 现在走在大街小巷都能看到二维码,而且最近由于项目需要,所以研究了下二维码开发的东西,开源的二维码扫描库主要有zxing和zbar,zbar在iPos平台上应用比较成熟,而在Android平台上主流还是用zxing库,因此这里主要讲述如何利用zxing
Android ListView的item背景色设置以及item点击无响应等相关问题 在Android开发中,listview控件是非常常用的控件,在大多数情况下,大家都会改掉listview的item默认的外观,下面讲解以下在使用listview时最常见的几个问题。1.如何改变item的背景色和按
如何向Android模拟器中导入含有中文名称的文件在进行Android开发的时候,如果需要向Android模拟器中导入文件进行测试,通过DDMS下手动导入或者在命令行下通过adb push命令是无法导入含有中文文件名的文件的。后来发现借用其他工具可以向模拟器中导入中文名称的文件,这个工具就是Ultr
Windows 下搭建Android开发环境一.下载并安装JDK版本要求JDK1.6+,下载JDK成功后进行安装,安装好后进行环境变量的配置【我的电脑】-——&gt;【属性】——&gt;【高级】 ——&gt;【环境变量】——&gt;【系统变量】中点击【新建】:变量名:CLASSPATH变量值:……
如何利用PopupWindow实现弹出菜单并解决焦点获取以及与软键盘冲突问题 在android中有时候可能要实现一个底部弹出菜单,此时可以考虑用PopupWindow来实现。下面就来介绍一下如何使用PopupWindow实现一个弹出窗。 主Activity代码:public void onCreat
解决Android中的ERROR: the user data image is used by another emulator. aborting的方法 今天调试代码的时候,突然出现这个错误,折腾了很久没有解决。最后在google上找到了大家给出的两种解决方案,下面给出这两种方法的链接博客:ht
AdvserView.java package com.earen.viewflipper; import android.content.Context; import android.graphics.Bitmap; import android.graphics.BitmapFactory;
ImageView的scaleType的属性有好几种,分别是matrix(默认)、center、centerCrop、centerInside、fitCenter、fitEnd、fitStart、fitXY。 |值|说明| |:--:|:--| |center|保持原图的大小,显示在ImageVie
文章浏览阅读8.8k次,点赞9次,收藏20次。本文操作环境:win10/Android studio 3.21.环境配置 在SDK Tools里选择 CMAKE/LLDB/NDK点击OK 安装这些插件. 2.创建CMakeLists.txt文件 在Project 目录下,右键app,点击新建File文件,命名为CMakeLists.txt点击OK,创建完毕! 3.配置文件 在CMa..._link c++ project with gradle
文章浏览阅读1.2w次,点赞15次,收藏69次。实现目的:由mainActivity界面跳转到otherActivity界面1.写好两个layout文件,activity_main.xml和otherxml.xmlactivity_main.xml&lt;?xml version="1.0" encoding="utf-8"?&gt;&lt;RelativeLayout ="http://schemas..._android studio 界面跳转
文章浏览阅读3.8w次。前言:最近在找Android上的全局代理软件来用,然后发现了这两款神作,都是外国的软件,而且都是开源的软件,因此把源码下载了下来,给有需要研究代理这方面的童鞋看看。不得不说,国外的开源精神十分浓,大家相互使用当前基础的开源软件,然后组合成一个更大更强的大开源软件。好吧,废话不多说,下面简单介绍一下这两款开源项目。一、ProxyDroid:ProxyDroid功能比较强大,用到的技术也比较多,源码也_proxydroid
文章浏览阅读2.5w次,点赞17次,收藏6次。创建项目后,运行项目时Gradle Build 窗口却显示错误:程序包R不存在通常情况下是不会出现这个错误的。我是怎么遇到这个错误的呢?第一次创建项目,company Domain我使用的是:aven.com,但是创建过程在卡在了Building 'Calculator' Gradle Project info这个过程中,于是我选择了“Cancel”第二次创建项目,我还是使用相同的项目名称和项目路_r不存在
文章浏览阅读8.9w次,点赞4次,收藏43次。前言:在Android上使用系统自带的代理,限制灰常大,仅支持系统自带的浏览器。这样像QQ、飞信、微博等这些单独的App都不能使用系统的代理。如何让所有软件都能正常代理呢?ProxyDroid这个软件能帮你解决!使用方法及步骤如下:一、推荐从Google Play下载ProxyDroid,目前最新版本是v2.6.6。二、对ProxyDroid进行配置(基本配置:) (1) Auto S_proxydroid使用教程
文章浏览阅读1.1w次,点赞4次,收藏17次。Android Studio提供了一个很实用的工具Android设备监视器(Android device monitor),该监视器中最常用的一个工具就是DDMS(Dalvik Debug Monitor Service),是 Android 开发环境中的Dalvik虚拟机调试监控服务。可以进行的操作有:为测试设备截屏,查看特定进程中正在运行的线程以及堆栈信息、Logcat、广播状态信息、模拟电话_安卓摄像头调试工具