我眼中的路由
提到路由我最先聯(lián)想到的是平時(shí)用來上網(wǎng)的無線路由器,而無線路由給我們帶來的好處在我看來有兩點(diǎn)。一羊壹、用戶不用關(guān)心無線路由是連接的網(wǎng)線、還是一個(gè)橋接的路由器齐婴,只需要使用賬號密碼即可上網(wǎng)油猫。二、我們更換上網(wǎng)方式比如撥號上網(wǎng)換個(gè)賬號尔店,只要保持路由器wifi名稱賬號密碼不變眨攘,用戶就可以不做任何更改繼續(xù)使用wifi。
在我看來路由就是一個(gè)映射規(guī)則嚣州,通過輸入得到輸出。就像無線路由器一樣輸入的是wifi的名稱密碼共螺,得到的是網(wǎng)絡(luò)數(shù)據(jù)该肴。我們定義好生成規(guī)則后就可以很簡單的從輸入得到輸出結(jié)果。
那移動開發(fā)中的路由是什么呢藐不?以iOS開發(fā)為例在我看來匀哄,就是一個(gè)根據(jù)規(guī)則生成控制器、視圖等的一個(gè)東西雏蛮。
1涎嚼、開發(fā)中使用路由的好處
個(gè)人理解和無線路由器好處類似。一挑秉、按照某種規(guī)則比如用鏈接和頁面建立對應(yīng)關(guān)系后法梯,我們通過鏈接就可以構(gòu)造出對應(yīng)的頁面、控件犀概,調(diào)用者和被調(diào)用者沒有直接依賴也不用關(guān)心具體的初始化步驟立哑。二、當(dāng)我們修改映射結(jié)果時(shí)姻灶,比如以前鏈接url對應(yīng)的是頁面A現(xiàn)在換成頁面B铛绰,調(diào)用的地方不用做任何修改,更加靈活产喉。提到路由往往就會講到組件化捂掰,路由是組件化中很重要的一部分但不在本篇文章的討論范圍,這里推薦一篇文章大家有興趣可以看看iOS 組件化方案探索曾沈。
2这嚣、我們項(xiàng)目的需求
①、通過h5晦譬、遠(yuǎn)程推送疤苹、公眾號等打開app跳轉(zhuǎn)到指定頁面。②敛腌、應(yīng)用內(nèi)有個(gè)任務(wù)系統(tǒng)卧土,可能跳轉(zhuǎn)到很多不同頁面惫皱。③、一些目標(biāo)頁需要登錄后方可進(jìn)入尤莺、一些目標(biāo)頁需要先出一個(gè)詢問彈窗點(diǎn)擊確定后才會跳轉(zhuǎn)旅敷。
3、頁面調(diào)用方式
①颤霎、app外調(diào)用分為三種遠(yuǎn)程推送媳谁、UniversalLink、URL Schemes友酱。②晴音、應(yīng)用內(nèi)調(diào)用
③、其中推送和應(yīng)用內(nèi)調(diào)用傳參客戶端可以隨意設(shè)定缔杉,主要是看deeplink和url schemes這兩種锤躁,而這兩種都是以鏈接的形式打開app的,所以我們就以鏈接來和頁面建立綁定關(guān)系或详。
4系羞、使用鏈接和頁面建立綁定關(guān)系
①、簡單介紹下鏈接的組成部分
例如:https://www.baidu.com/s?inputT=3358&rsv_sug4=3358
scheme(https)霸琴、host(www.baidu.com)椒振、path(/s)、參數(shù)(?inputT=3358&rsv_sug4=3358)
②梧乘、首先建立綁定關(guān)系
由于使用deeplink打開使用的是https澎迎、使用urlscheme使用的是自己定義的一個(gè)字符串,所以scheme是不固定的宋下。每個(gè)鏈接參數(shù)肯定也是不固定的嗡善,所以我們選用host+path來作為key對應(yīng)具體的頁面。
protocol AIRouterProtocol {
static func targetWith(pa: [String: Any]) -> AIRouterProtocol?
func needLogin() -> Bool
func isPush() -> Bool
}
var targetDict = [String: AIRouterProtocol.Type]()
func registerRouter(target: AIRouterProtocol.Type, key: String) {
targetDict.updateValue(target, forKey: key)
}
如上面代碼所示我們將頁面和鏈接的映射關(guān)系存儲在了一個(gè)字典里学歧,以連接的host+path為key罩引,以一個(gè)遵從我們定義的路由協(xié)議為value。協(xié)議主要定義了三個(gè)方法枝笨,targetWith(頁面的構(gòu)造方法)袁铐、needLogin(頁面是否需要登錄)、isPush(頁面跳轉(zhuǎn)方式)横浑。
5剔桨、通過鏈接獲取頁面
①、獲取鏈接各個(gè)組成部分徙融,取出host+path作為key值獲取對應(yīng)頁面洒缀,取出鏈接中的參數(shù)部分初始化頁面。考慮到應(yīng)用內(nèi)使用時(shí)傳遞一些鏈接無法傳遞的參數(shù)類型树绩,如block萨脑、UIImage等,提供了一個(gè)externParameter字典類型參數(shù)和鏈接里的參數(shù)共同組成參數(shù)部分來初始化頁面饺饭。
func targetWith(urlStr: String, externParameter: [String: Any]? = nil) -> AIRouterProtocol? {
let encodeUrlStr = urlStr.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed) ?? ""
if let urlComponents = URLComponents(string: encodeUrlStr) {
let scheme = urlComponents.scheme ?? ""
let host = urlComponents.host ?? ""
let path = urlComponents.path
//AILog("scheme:\(scheme) host:\(host) path:\(path)")
var parameter = [String: Any]()
if let queryItems = urlComponents.queryItems {
for query in queryItems {
parameter.updateValue(query.value ?? "", forKey: query.name)
}
}
if let externDic = externParameter {
for (key, value) in externDic {
parameter.updateValue(value, forKey: key)
}
}
if scheme == kAppScheme {
return targetWith(key: host + path, parameter: parameter)
} else if kHttp.contains(scheme) {
if let target = targetWith(key: host + path, parameter: parameter) {
return target
}
}
}
return nil
}
func targetWith(key: String, parameter: [String: Any]) -> AIRouterProtocol? {
if let router = targetDict[key] {
return router.targetWith(pa: parameter)
}
return nil
}
6渤早、調(diào)用
通過鏈接初始化頁面,然后通過協(xié)議約定的方法獲取是否需要登錄瘫俊、跳轉(zhuǎn)方式完成跳轉(zhuǎn)鹊杖。(如果不需跳轉(zhuǎn)初始化方法返回nil即可,然后做自己想做的事兒扛芽,如展示一個(gè)彈窗骂蓖、tabbar切換選中tab等)
/// 處理鏈接(打開頁面/其它處理)
/// - Parameter urlStr: 鏈接
/// - Parameter externParameter: 額外參數(shù)(一些參數(shù)無法放在鏈接中如block、UIImage等可以放在這里)
static func openUrl(urlStr: String, externParameter: [String: Any]? = nil) {
if let target = AIRouter.share.targetWith(urlStr: urlStr, externParameter: externParameter) {
let needLogin = target.needLogin()
let isPush = target.isPush()
if let vc = target as? UIViewController {
self.openVC(vc: vc, needLogin: needLogin, isPush: isPush)
}
}
}
static func openVC(vc: UIViewController, needLogin: Bool, isPush: Bool) {
if let topVC = UIViewController.topViewController() {
if needLogin && UserManager.share.UserIsLogin == false {//登錄處理
let loginVC = LoginViewController {
self.openVC(vc: vc, needLogin: needLogin, isPush: isPush)
}
self.openVC(vc: loginVC, needLogin: false, isPush: true)
} else {
if isPush {
if let _ = topVC.navigationController {
topVC.aiPushToVC(toVC: vc)
} else {
let navi = UINavigationController(rootViewController: vc)
topVC.aiPresent(navi, animated: true, completion: nil)
}
} else {
topVC.aiPresent(vc, animated: true, completion: nil)
}
}
}
}