本文记录了与一位同学关于Router设计的争论, 对于url router 和 protocol router 的争论, 架构并没有孰优孰劣, 只有适合与否, 希望能有更多的同学一起踊跃探讨.
对于组件化, 相信大家一定不陌生, 但针对组件化的方案及思路, 大家或多或少都有一些自己的想法, 如果不清楚组件化的同学可以先通过以下文章预习一下.
alibaba/BeeHive
蘑菇街 App 的组件化之路
蘑菇街 App 的组件化之路·续
iOS应用架构谈 组件化方案
iOS组件化方案
组件化架构漫谈
iOS 组件化 —— 路由设计思路分析
iOS VIPER架构实践(三):面向接口的路由设计
Hybird 搭建路由Router实现组件化
一切源于我将 -- 架构初探系列 -- 发布到掘金上和一位大佬对于Router设计展开的激烈的辩论, 作为一个iOS初级玩家, 觉得应该将争辩的内容分享出来让大家一起探讨才能够有所提升.
以下是我对于Router的简单设计, 这个在之前的文章里也有写到.
- //
- // Router.swift
- // RouterPatterm
- //
- // Created by 双泉 朱 on 17/4/12.
- // Copyright © 2017年 Doubles_Z. All rights reserved.
- //
- import UIKit
- class Router {
- static let shareRouter = Router()
- var params: [String : Any]?
- var routers: [String : Any]?
- fileprivate let map = ["J1" : "Controller"]
- func guardRouters(finishedCallback : @escaping () -> ()) {
- Http.requestData(.get, URLString: "http://localhost:3001/api/J1/getRouters") { (response) in
- guard let result = response as? [String : Any] else { return }
- guard let data:[String : Any] = result["data"] as? [String : Any] else { return }
- guard let routers:[String : Any] = data["routers"] as? [String : Any] else { return }
- self.routers = routers
- finishedCallback()
- }
- }
- }
- extension Router {
- func addParam(key: String, value: Any) {
- params?[key] = value
- }
- func clearParams() {
- params?.removeAll()
- }
- func push(_ path: String) {
- guardRouters {
- guard let state = self.routers?[path] as? String else { return }
- if state == "app" {
- guard let nativeController = NSClassFromString("RouterPatterm.\(self.map[path]!)") as? UIViewController.Type else { return }
- currentController?.navigationController?.pushViewController(nativeController.init(), animated: true)
- }
- if state == "web" {
- let host = "http://localhost:3000/"
- var query = ""
- let ref = "client=app"
- guard let params = self.params else { return }
- for (key, value) in params {
- query += "\(key)=\(value)&"
- }
- self.clearParams()
- let webViewController = WebViewController("\(host)\(path)?\(query)\(ref)")
- currentController?.navigationController?.pushViewController(webViewController, animated: true)
- }
- }
- }
- }
使用的时候只需要如下操作:
- Router.shareRouter.params = [
- "text" : "app端 传入数据",
- "code" : 1001
- ]
- Router.shareRouter.push("J1")
- override func viewDidLoad() {
- super.viewDidLoad()
- self.text = Router.shareRouter.params["text"]
- }
以上就是我理解router的本质, 最为精简的功能, 可选添加参数, AOP清除参数, 跳转. 当然要路由是要根据业务来设计的, 这里只是实现一个最基础的功能.
接着就有一位同学@黑超熊猫zuik在我的文章后进行了评论, @黑超熊猫zuik是 iOS Developer专栏的管理员(有30k+关注), 之后我们就router的 map 和 protocol 展开了交流, 以下就是交流的过程.
顽主范儿
请教一下,Router的push方法怎么往目标viewController中传递参数
Castie1
回复顽主范儿:我这个router设计的比较简单, 作为一个单例持有一个参数字典, 如果需要传递参数的时候调用 addParam, 因为并不是所有跳转都需要传参数的, 使用aop在目的页面hook生命周期删除参数字典就可以了. 当然router如何设计很大程度与公司项目制定的需求及规范有关, 所以这块没有办法脱离业务来谈, 只能描述下本质.
顽主范儿
回复Castie1:我们之前也考虑过组件化,但是考虑到这个map还有整个架构都依赖于字符串,记得还有其他一些局限性,看了链家蘑菇街等分享的方案,依然没有解决这些问题,所以最后放弃了ᕙ(⇀‸↼‵‵)ᕗ
Castie1
回复顽主范儿:依赖于字符串并不是什么坏事, 因为字符串错误的问题在开发环境上就会解决了, 使用extern也可以减少这种事情的发生, casatwy.com/iOS-Moduliz… 这篇文章的思想就是完全的去模型化, 可能比较激进, 但供你参考.
顽主范儿 回复Castie1:多谢,之前也看过这篇文章,由于我们没有服务器配置router的需求,只是单纯的想把各个大模块解耦,这种情况下考虑到组件化带来的利并没有特别大于弊,所以还是没有采用,再次感谢
来源: https://juejin.im/entry/5a24bc6d518825619a02836f