
 假设现在有两个业务组件登录和问答模块之间需要进行通信,可能会想到用反射的方式,是可以但是会影响性能,而写的代码比较多类名这些要记清楚。
 路由可以看做表,每个map对应一张表
 我们可以试着这么写,完成MainActivity跳转到LoginActivity,LoginActivity再跳转到WendaActivity。
 新建一个libRouter的library
 
 新建一个Router类进行路由表的注册和启动
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
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
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 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注册路由表
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这个方法其实也就是对应下面的方法。
 



















