Android AutoService 组件化

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: c a l l B a c k N a m e , R e s p o n s e : callBackName,Response: 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 的功能、接口使用;
粉丝技术交流裙

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

尾声

评论里面有些同学有疑问关于如何学习material design控件,我的建议是去GitHub搜,有很多同行给的例子,这些栗子足够入门。

有朋友说要是动真格的话,需要NDK以及JVM等的知识,首现**NDK并不是神秘的东西,**你跟着官方的步骤走一遍就知道什么回事了,无非就是一些代码格式以及原生/JAVA内存交互,进阶一点的有原生/JAVA线程交互,线程交互确实有点蛋疼,但平常避免用就好了,再说对于初学者来说关心NDK干嘛,据鄙人以前的经历,只在音视频通信和一个嵌入式信号处理(离线)的两个项目中用过,嵌入式信号处理是JAVA->NDK->.SO->MATLAB这样调用的我原来MATLAB的代码,其他的大多就用在游戏上了吧,一般的互联网公司会有人给你公司的SO包的。
至于JVM,该掌握的那部分,相信我,你会掌握的,不该你掌握的,有那些专门研究JVM的人来做,不如省省心有空看看计算机系统,编译原理。

一句话,平常多写多练,这是最基本的程序员的素质,尽量挤时间,读理论基础书籍,JVM不是未来30年唯一的虚拟机,JAVA也不一定再风靡未来30年工业界,其他的系统和语言也会雨后春笋冒出来,但你理论扎实会让你很快理解学会一个语言或者框架,你平常写的多会让你很快熟练的将新学的东西应用到实际中。
初学者,一句话,多练。

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

是JAVA->NDK->.SO->MATLAB这样调用的我原来MATLAB的代码,其他的大多就用在游戏上了吧,一般的互联网公司会有人给你公司的SO包的。**
至于JVM,该掌握的那部分,相信我,你会掌握的,不该你掌握的,有那些专门研究JVM的人来做,不如省省心有空看看计算机系统,编译原理。

一句话,平常多写多练,这是最基本的程序员的素质,尽量挤时间,读理论基础书籍,JVM不是未来30年唯一的虚拟机,JAVA也不一定再风靡未来30年工业界,其他的系统和语言也会雨后春笋冒出来,但你理论扎实会让你很快理解学会一个语言或者框架,你平常写的多会让你很快熟练的将新学的东西应用到实际中。
初学者,一句话,多练。

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

  • 21
    点赞
  • 30
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值