激情久久久_欧美视频区_成人av免费_不卡视频一二三区_欧美精品在欧美一区二区少妇_欧美一区二区三区的

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務(wù)器之家 - 編程語言 - Android - 用Kotlin打造一個(gè)Router的示例代碼

用Kotlin打造一個(gè)Router的示例代碼

2022-01-24 15:58滑板上的老砒霜 Android

本篇文章主要介紹了用Kotlin打造一個(gè)Router的示例代碼,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧

0.

最近接收了從前的項(xiàng)目,之前的代碼比較凌亂,準(zhǔn)備重構(gòu)。整個(gè)項(xiàng)目其實(shí)功能比較明顯,而且產(chǎn)品還想給他打造成比較成熟牛逼的app,那么組件化開發(fā)勢(shì)在必行。眾所周知,模塊開發(fā),路由先行,就有了這篇文章。自從用了Kotlin后,感覺Java那些繁瑣的語法和喋喋不休的空指針判斷讓人惡心,于是決定用Kotlin來寫一個(gè)Router

1.

項(xiàng)目地址Router

代碼,想好再寫,首先分析一下為什么需要模塊化開發(fā),為什么路由并且這個(gè)路由需要什么功能。

1.1

為什么使用模塊化開發(fā):隨著項(xiàng)目的越來越大,如果把所有功能都放到一個(gè)module里,對(duì)開發(fā)和測(cè)試來說都有一個(gè)效率問題,對(duì)開發(fā)來講,由于各種業(yè)務(wù)代碼混合在一起,出現(xiàn)問題不好定位,對(duì)測(cè)試來講,每次修改都要重新編譯整個(gè)項(xiàng)目測(cè)試。采用組件化開發(fā),講業(yè)務(wù)模塊分到一個(gè)一個(gè)module里,彼此間相互獨(dú)立,這樣既容易定位問題,也方便測(cè)試人員進(jìn)行測(cè)試,因?yàn)橹恍枰獪y(cè)試相應(yīng)的module即可。所以我們的項(xiàng)目結(jié)構(gòu)應(yīng)該是這樣的

用Kotlin打造一個(gè)Router的示例代碼

1.2

為什么使用路由:上面說了,每個(gè)module是彼此獨(dú)立的,而要實(shí)現(xiàn)activity的跳轉(zhuǎn)就需要彼此引用,這是我們不想看到的,模塊間應(yīng)該是沒有依賴的,那么如何實(shí)現(xiàn)不同模塊的跳轉(zhuǎn)呢?就需要路由了。

1.3

這個(gè)路由需要什么功能:目前所需要的就是根據(jù)url實(shí)現(xiàn)不同模塊之間的Activity跳轉(zhuǎn),包括傳遞參數(shù),類似startActivity和startActivityForResult,得到不同模塊的fragment。

2.

需求想好了,該想怎么實(shí)現(xiàn)了。activity跳轉(zhuǎn)有兩種,隱式調(diào)用和顯示調(diào)用,如果采取隱式調(diào)用就要為每個(gè)activity注冊(cè)intent-filter,麻煩,那就只能顯示調(diào)用了,但是顯示調(diào)用是需要class對(duì)象的,不同module是不能得到class對(duì)象的,這怎么搞?編譯前得不到,運(yùn)行時(shí)總行吧,所以我們需要一個(gè)容器來存儲(chǔ)不同moudle里的class對(duì)象,并且能通過一個(gè)值來進(jìn)行區(qū)分,所以我們需要一個(gè)map,而值就是用來區(qū)分各個(gè)activity和fragment的url。

第二個(gè)問題來了,url如何確定,class對(duì)象怎么得到,又怎么放進(jìn)map里。這里我們采用注解來做,我們用在注解里聲明了url,同時(shí)自定義Processor,在編譯時(shí)生成java文件,里面只有一個(gè)方法

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

根據(jù)講每個(gè)module里添加注解的Activity和fragment的class對(duì)象放入傳入的map里。之后再Application的onCreate方法里,調(diào)用Router的register方法

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
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()
      }
    }
 
  }

這個(gè)方法很簡(jiǎn)單,調(diào)用每個(gè)module里利用Processor生成的對(duì)象的putRouteClass方法,將Router里的一個(gè)全局map傳入,這樣,這個(gè)map就持有了所有添加注解的Activity和fragment的url以及對(duì)應(yīng)的class對(duì)象。有了class對(duì)象,那想怎么搞就怎么搞了

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
//類似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

注冊(cè)注解解釋器的時(shí)候,不要使用google的autoservice庫(kù)了,采用resoureces,META-INF,不然沒效果。別問為什么,我也不知道

3.2

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

?
1
2
3
4
5
apply plugin: 'kotlin-kapt'
dependencies {
kapt project(':processor')
api project(':router')
}

3.3

生成的java文件在每個(gè)module的build/generated/source/kapt里

4.

其實(shí)在上家公司的時(shí)候就打算寫一個(gè)路由,只是由于種種原因最近沒能成型,這里只是給大家提供一種思路,作為思路文,就不在普及注解和編譯時(shí)注解解釋器了,請(qǐng)自行查詢資料。

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持服務(wù)器之家。

原文鏈接:https://www.jianshu.com/p/185deb9f8615

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 91久久国产综合久久91猫猫 | 中文字幕视频在线播放 | 久久精品视频1 | 亚洲欧美国产高清va在线播放 | 国产日韩中文字幕 | av在线看网站| 久久久精品视频国产 | 在线亚洲欧美日韩 | 免费人成在线播放 | 一级毛片免费版 | av在线免费观看国产 | 日本在线播放一区二区三区 | 美女黄色毛片免费看 | 国产乱xxxx | 欧美一区二区三区久久精品视 | 国产精品岛国久久久久久久 | 久色精品 | 国产亚洲精品一区二区三区 | 久久国产不卡 | 日本在线免费观看视频 | 羞羞电影在线观看www | 少妇一级淫片高潮流水电影 | 视频国产一区二区 | 国产伊人色 | 操碰视频在线观看 | 牛牛碰在线 | 亚洲一区二区三区日本久久九 | 国产四区 | 一级做受大片免费视频 | 午夜影视一区二区 | 麻豆传传媒久久久爱 | 一区在线视频 | av观看国产 | 日本黄色大片免费 | 日韩毛片免费观看 | 国产高潮国产高潮久久久91 | 免费永久在线观看黄网 | 中文字幕亚洲一区二区三区 | 成人毛片在线免费观看 | 一本在线高清码电影 | 日本在线免费观看视频 |