亲宝软件园·资讯

展开

Kotlin Router 用Kotlin打造一个Router的代码实例

滑板上的老砒霜 人气:0
想了解用Kotlin打造一个Router的代码实例的相关内容吗,滑板上的老砒霜在本文为您仔细讲解Kotlin Router的相关知识和一些Code实例,欢迎阅读和指正,我们先划重点:Kotlin,Router,下面大家一起来学习吧。

0.

最近接收了从前的项目,之前的代码比较凌乱,准备重构。整个项目其实功能比较明显,而且产品还想给他打造成比较成熟牛逼的app,那么组件化开发势在必行。众所周知,模块开发,路由先行,就有了这篇文章。自从用了Kotlin后,感觉Java那些繁琐的语法和喋喋不休的空指针判断让人恶心,于是决定用Kotlin来写一个Router

1.

项目地址Router

代码,想好再写,首先分析一下为什么需要模块化开发,为什么路由并且这个路由需要什么功能。

1.1

为什么使用模块化开发:随着项目的越来越大,如果把所有功能都放到一个module里,对开发和测试来说都有一个效率问题,对开发来讲,由于各种业务代码混合在一起,出现问题不好定位,对测试来讲,每次修改都要重新编译整个项目测试。采用组件化开发,讲业务模块分到一个一个module里,彼此间相互独立,这样既容易定位问题,也方便测试人员进行测试,因为只需要测试相应的module即可。所以我们的项目结构应该是这样的

1.2

为什么使用路由:上面说了,每个module是彼此独立的,而要实现activity的跳转就需要彼此引用,这是我们不想看到的,模块间应该是没有依赖的,那么如何实现不同模块的跳转呢?就需要路由了。

1.3

这个路由需要什么功能:目前所需要的就是根据url实现不同模块之间的Activity跳转,包括传递参数,类似startActivity和startActivityForResult,得到不同模块的fragment。

2.

需求想好了,该想怎么实现了。activity跳转有两种,隐式调用和显示调用,如果采取隐式调用就要为每个activity注册intent-filter,麻烦,那就只能显示调用了,但是显示调用是需要class对象的,不同module是不能得到class对象的,这怎么搞?编译前得不到,运行时总行吧,所以我们需要一个容器来存储不同moudle里的class对象,并且能通过一个值来进行区分,所以我们需要一个map,而值就是用来区分各个activity和fragment的url。

第二个问题来了,url如何确定,class对象怎么得到,又怎么放进map里。这里我们采用注解来做,我们用在注解里声明了url,同时自定义Processor,在编译时生成java文件,里面只有一个方法

public void putRouteClass(ArrayMap<String, Class<?>> routableMap) {
routableMap.put("test", MainActivity.class);
}

根据讲每个module里添加注解的Activity和fragment的class对象放入传入的map里。之后再Application的onCreate方法里,调用Router的register方法

fun register(vararg moduleNames: String)
  {
    for (moduleName in moduleNames)
    {
      try
      {
        val cla = Class.forName(Constants.PACKAGE_NAME + Constants.DOT + moduleName + "_" + Constants.ROUTER_TABLE_IMP)

        val routerTable = cla.newInstance() as RouterTable

        routerTable.putRouteClass(classMap)
      } catch (e: ClassNotFoundException)
      {
        e.printStackTrace()
      } catch (e: Exception)
      {
        e.printStackTrace()
      }
    }

  }

这个方法很简单,调用每个module里利用Processor生成的对象的putRouteClass方法,将Router里的一个全局map传入,这样,这个map就持有了所有添加注解的Activity和fragment的url以及对应的class对象。有了class对象,那想怎么搞就怎么搞了

//类似startActivity
fun go(context: Context, url: String, extras: Bundle? = null)
  {
    val intent = Intent(context, classMap[url])

    if (extras != null)
    {
      intent.putExtras(extras)
    }

    context.startActivity(intent)
  }

  fun go(fragment: Fragment,url: String, extras: Bundle? = null)
  {
    val context=fragment.context
    if(context!=null)
    {
      go(context, url, extras)
    }
  }
  //类似startActivityForResult
  fun goForResult(context: Context, url: String, requestCode: Int, extras: Bundle? = null)
  {
    val intent = Intent(context, classMap[url])

    if (extras != null)
    {
      intent.putExtras(extras)
    }

    if (context is Activity)
    {
      context.startActivityForResult(intent, requestCode)
    } else if (context is Fragment)
    {
      context.startActivityForResult(intent, requestCode)
    }
  }

 //得到fragment
  fun getFragment(url: String): Fragment?
  {
    try
    {
      val cla = classMap[url]

      if (cla != null)
      {
        return cla.newInstance() as Fragment
      } else
      {

      }

    } catch (e: ClassNotFoundException)
    {
      e.printStackTrace()
    } catch (e: Exception)
    {
      e.printStackTrace()
    }
    return null
  }

3.有些坑:

3.1

注册注解解释器的时候,不要使用google的autoservice库了,采用resoureces,META-INF,不然没效果。别问为什么,我也不知道

3.2

在gradle文件里使用注解解释器使用kapt代替annoationProcessor

apply plugin: 'kotlin-kapt'
dependencies {
kapt project(':processor')
api project(':router')
}

3.3

生成的java文件在每个module的build/generated/source/kapt里

4.

其实在上家公司的时候就打算写一个路由,只是由于种种原因最近没能成型,这里只是给大家提供一种思路,作为思路文,就不在普及注解和编译时注解解释器了,请自行查询资料。

加载全部内容

相关教程
猜你喜欢
用户评论