用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.

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值