iOS(Swift) 路由設(shè)計二

2023-02-26 今天難得的一個好天氣,一整天陽光暖洋洋,精力十足,整理框架時,對路由一些小想法,于是花了一整天,把整個路由組件做成了pod,抽了出來.

上一篇路由講了個當時隨手寫的一個框架,但寫完后項目沒怎么用,后面因為事情太多,漸漸也就忘了,這次重新整理了一下思路,對路由重新設(shè)計了一番.

架構(gòu)

|____Classes
| |____GetRouter.swift
| |____GetRouterHandlerSource.swift
| |____.gitkeep
| |____ReplaceMe.swift
| |____GetRouterMiddleware.swift
| |____GetRouterName.swift
| |____GetRouterHandler.swift
|____Assets
| |____.gitkeep

路由設(shè)計

路由分遠程路由和本地路由,我們需要做的是統(tǒng)一,遠程路由適用與banner跳轉(zhuǎn),js交互操作等等情況,一般來說是string執(zhí)行的路由跳轉(zhuǎn),本地路由需要規(guī)定要,一個頁面的跳轉(zhuǎn),最好走唯一的路由,好處就是封裝,低耦合,A頁面理論上是不應(yīng)該import B頁面的,還有就是埋點參數(shù)上報等等,統(tǒng)一路徑.

流程

  • 遠程路由
    urlStirng->encoding->parse解析路由,合并參數(shù)->匹配本地路由->執(zhí)行跳轉(zhuǎn)
  • 本地路由
    匹配本地路由->執(zhí)行跳轉(zhuǎn)

GetRouter

遠程路由與本地路由解析的流程文件后,解析完成后交給GetRougerHandler執(zhí)行

GetRougerHandler

匹配路由名相同的GetPage 執(zhí)行中間件攔截, 執(zhí)行跳轉(zhuǎn)

GetRouterMiddleware中間件攔截

中間件用于公共攔截方式,舉例說明,我們?nèi)绻胩D(zhuǎn)vip直播間,我們需要前置判斷當前登錄者是不是vip,當前vip直播間是否正在直播,一個是同步就可以判斷出當前用戶vip狀態(tài),一個需要實時請求后端直播狀態(tài)接口等.
傳統(tǒng)方式中,我們可能需要通用寫個判斷

// 判斷vip
guard User.isVip else {
return 
}
// 判斷直播狀態(tài)
RequestLiveStatus {
  jumpLive()
}

這樣寫不太優(yōu)雅, 所以我增加了同步攔截與異步攔截(注意,中間件的環(huán)境在iOS13以上處于異步線程,執(zhí)行UI操作,吐絲重定向等請回到主線程執(zhí)行)

open class GetRouterMiddleware {
    /// 優(yōu)先級
    let priority: Int
    
    public init(priority: Int) {
        self.priority = priority
    }
    /// 處理返回結(jié)果
    /// - Returns: true: 允許通過 false: 禁止
    open func handler(routeName: GetRouterName, params: GetDict?) -> Bool { true }
    
    @available(iOS 13.0.0, *)
    /// 異步中間件, 注意如果想要在此出重定向路由,注意要回到mainThread
    /// - Parameter routeName: 路由名
    /// - Returns: true: 允許通過 false: 禁止
    open func handlerAsync(routeName: GetRouterName, params: GetDict?) async throws -> Bool { true }
}

這里注意一下,異步攔截是iOS13以上可用,需要自行構(gòu)造一個async任務(wù)(相關(guān)只是可以參考await async 構(gòu)造async),demo也有相關(guān)示例,使用起來也很方便

GetPage

注冊路由時包裝的路由匹配信息,包括中間件和行為.

GetRouterHandlerSource

提供GetPage的source來源

GetRouterName路由名設(shè)計

原先我們設(shè)計的路由名是這樣的

enum GetRouterName: String {
    // MARK: - --------------------------------------公共
    case http = "http"
    case https = "https"
    // MARK: - --------------------------------------發(fā)現(xiàn)
    
    // MARK: - --------------------------------------社區(qū)
    case community_post_detail = "community_post_detail"
    // MARK: - --------------------------------------個人
    
    // MARK: - --------------------------------------我的
    case mine_setting = "mine_setting"
    case mine_complete = "mine_complete"
    // MARK: - --------------------------------------帖子
    case post_detail = "post_detail"
}

很簡單,很易懂,但是有缺點

  • enum可以在switch中可以枚舉掉所有的選項,保證我們不會漏掉任何路由名,但是,enum無法寫成extension,也就是說,我們必須把所有名卸載一個文件里,這樣對模塊化是不好的,我希望使用的場景是我每個模塊去注冊這個路由,然后提供我模塊的模塊名和路由處理選項,這樣就可以做到單獨開發(fā)自己的選項
  • string不是類型,在本地調(diào)用中,我們不可避免的會出現(xiàn)必須強制帶類名的方式,寫起來就不是那么清爽,例如Router.to(GetRouterName.mine_setting),我想要的是Router.to(.mine_setting)
    所以,利用swift的語法特點,我修改成這種寫法
/// 路由列表容器
public struct GetRouterNames {}

/// 路由列表
public struct GetRouterName: RawRepresentable,
                      ExpressibleByStringLiteral,
                      Hashable {
    
    public var rawValue: String
    
    public init(rawValue: String) {
        self.rawValue = rawValue
    }
    
    public init(stringLiteral value: StringLiteralType) {
        self.rawValue = value
    }
    
    public var hashValue: Int {
        rawValue.hashValue
    }
}

這里寫了一個類型GetRouterName,利用swift字面量,可以達到這種定義路由名的效果

//  GetRouter+Home.swift
//  quick
//
//  Created by suyikun on 2023/2/25.
//

import Foundation

extension GetRouterNames {
    /// 發(fā)現(xiàn)列表
    static let discover: GetRouterName = "/discover/list"
    /// 發(fā)現(xiàn)頁詳情
    static let discoverDetail: GetRouterName = "/discover/detail"
}

雖然,我們損失了enum強制枚舉所有枚舉的效果,但是,我們收獲了模塊化路由名+路由名類型的優(yōu)點

使用示例




注意事項: 獲取參數(shù)時,遠程路由獲取的value是string類型,需要在寫代碼時考慮兼容轉(zhuǎn)換

自己在業(yè)務(wù)跳轉(zhuǎn)方面,做好兼容即可, 例如下圖這樣處理

PS

在我看來,路由設(shè)計應(yīng)該是可以純粹的解耦合的,他跟業(yè)務(wù)的跳轉(zhuǎn)沒有一點關(guān)系,只是一些規(guī)則的解析,
總結(jié)一下,回顧了下整個路由設(shè)計,一天的收獲滿滿,如果你們喜歡我的文章,或者對路由設(shè)計有什么批評建議意見,希望可以點贊留言
最后留一下git地址: 戳這前往

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,156評論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,401評論 3 415
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,069評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,873評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,635評論 6 408
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,128評論 1 323
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,203評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,365評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,881評論 1 334
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 40,733評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,935評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,475評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,172評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,582評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,821評論 1 282
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,595評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,908評論 2 372

推薦閱讀更多精彩內(nèi)容