假设现在有两个业务组件登录和问答模块之间需要进行通信,可能会想到用反射的方式,是可以但是会影响性能,而写的代码比较多类名这些要记清楚。
路由可以看做表,每个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=pathvar group:HashMap<String,Class<*>>?=nullif(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=pathvar group:HashMap<String,Class<*>>?=nullif(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"><TextViewandroid: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这个方法其实也就是对应下面的方法。