Android组件化中的Arouter学习

假设现在有两个业务组件登录和问答模块之间需要进行通信,可能会想到用反射的方式,是可以但是会影响性能,而写的代码比较多类名这些要记清楚。

路由可以看做表,每个map对应一张表

我们可以试着这么写,完成MainActivity跳转到LoginActivity,LoginActivity再跳转到WendaActivity。

新建一个libRouter的library

新建一个Router类进行路由表的注册和启动

kotlin 复制代码
class Router private constructor() {

    //先通过组名找到map再通过map找到对应的类名
    final val groupMap: HashMap<String, HashMap<String, Class<*>>> = HashMap()
    //最终将LoginActivity,WendaActivity注册到routeMap里面去
    final val routeMap:HashMap<String,Class<*>> =HashMap()

    object Holder {
        var INSTANCE = Router()
    }

    fun getInstance(): Router {
        return Holder.INSTANCE
    }

    //注册
    fun register(path:String,clz:Class<*>){
        val strArray: Array<String> = path.split("/").toTypedArray()
        if(strArray.size>2){
            var groupName:String=strArray[1]
            var routeName=path
            var group:HashMap<String,Class<*>>?=null
            if(groupMap.containsKey(groupName)){
                group=groupMap.get(groupName)
            }
            if(group==null){
                group= HashMap()
                groupMap.put(groupName,group)
            }
            if(group!=null){
                group.put(routeName,clz)
            }
        }
    }
    //启动
    fun startActivity(activity:Activity,path: String){
        val strArray: Array<String> = path.split("/").toTypedArray()
        if(strArray.size>2){
            var groupName:String=strArray[1]
            var routeName=path
            var group:HashMap<String,Class<*>>?=null
            if(groupMap.containsKey(groupName)){
                group=groupMap.get(groupName)
            }
            if(group!=null && group.containsKey(routeName)){
                val clz=group.get(routeName)
                activity.startActivity(Intent(activity,clz))
            }
        }
    }


}

MainActivity跳转LoginActivity

kotlin 复制代码
class MainActivity : AppCompatActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        Router.Holder.INSTANCE.startActivity(this,"/login/LoginActivity")
    }
}

LoginActivity跳转WendaActivity

kotlin 复制代码
class LoginActivity : AppCompatActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_login)
    }

    fun onClickView(view: View) {


      Router.Holder.INSTANCE.startActivity(this,"/wenda/WendaActivity")

    }
}

loginActivity布局增加点击事件

xml 复制代码
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".LoginActivity">

    <TextView
        android:onClick="onClickView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="LoginActivity"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

</androidx.constraintlayout.widget.ConstraintLayout>

Application注册路由表

kotlin 复制代码
class SunBeachApplication:Application() {


    override fun onCreate() {
        super.onCreate()
        Router.Holder.INSTANCE.register("/login/LoginActivity",LoginActivity::class.java)
        Router.Holder.INSTANCE.register("/wenda/WendaActivity",WendaActivity::class.java)

    }
}

运行app从MainActivity跳转到LoginActivity

点击LoginActivity跳转到WendaActivity

但也有问题并且如果要跳转的Activity比较多,Application里的onCreate方法需要注册比较多

如果我们在Router类进行添加的话就会出现无法识别出几个类的问题

所以我們需要一个注解处理器,由Activity去扫描,扫描到添加进来。

这个时候我们再来看一下Arouter是怎么实现的。

Arouter核心原理:

当我们使用Arouter的注解处理器的时候,就会帮我们生成相应的类,Arouter$$Group login就是login的group,而Arouter Root login就是去查找到group并调用group的方法

可以看到Arouter Root login通过传递routes将Arouter Group login的class添加到Map中。

而Arouter Group login里当调用了loadInto方法的话就会将LoginActivity注册进来

将相应的信息放到RouteMeta当中,RouteMeta就是一个包装类,封装了当前的一些信息,主要通过RouteMeta找到对应的class进行跳转或者参数传递。

当我们在使用Arouter的时候都需要进行init()初始化

这个_ARouter就是路由控制器。

这里主要是获取之前保存的一些信息,而不用for循环去遍历,这样下次可以直接用。

第一次通过getFileNameByPackageName把指定包下面的所有文件找出来,也就是生成的下面包的文件

也指定了要找对应包名的文件

并返回到routerMap的集合里去

然后再去遍历routerMap

而当第二次,缓存到本地后,直接读取拿到信息,就不用再去找。


这个时候在遍历routerMap的时候就会拿到这个root文件去调用loadInto方法

loadInto方法的具体实现也就是生成的那些类中

它会找到生成的group直接put到分组当中,分组当中提供了一个方法

通过这个方法可以将path,class这些信息放到一个map当中。因为生成文件中有很多group,有grouplogin,groupmain,要找到这些group调用loadInto方法,这样我们通过传入path路径,就能找到RouteMeta,就可以拿到class。

而当我们要去跳转,这个时候会进行build操作

会返回一个Postcard对象里面也包含了一些信息

接着继续点进去发现navigation这个方法调用了LogisticsCenter.completion(postcard)

调用loadInto这个方法其实也就是对应下面的方法。

相关推荐
Sinclair12 小时前
简单几步,安卓手机秒变服务器,安装 CMS 程序
android·服务器
雮尘15 小时前
手把手带你玩转Android gRPC:一篇搞定原理、配置与客户端开发
android·前端·grpc
ktl17 小时前
Android 编译加速/优化 80%:一个文件搞定,零侵入零配置
android
alexhilton1 天前
使用FunctionGemma进行设备端函数调用
android·kotlin·android jetpack
冬奇Lab1 天前
InputManagerService:输入事件分发与ANR机制
android·源码阅读
张小潇1 天前
AOSP15 Input专题InputManager源码分析
android·操作系统
RdoZam1 天前
Android-封装基类Activity\Fragment,从0到1记录
android·kotlin
奥陌陌2 天前
android 打印函数调用堆栈
android
用户985120035832 天前
Compose Navigation 3 深度解析(二):基础用法
android·android jetpack
恋猫de小郭2 天前
Android 官方正式官宣 AI 支持 AppFunctions ,Android 官方 MCP 和系统级 OpenClaw 雏形
android·前端·flutter