URLNavigator是Swift版本的Router。
Router的主要作用是解耦。之前在各个ViewController间跳转,需要import ViewController,这样就造成ViewController之间的依赖,也即耦合。通过router不需要再import ViewController。所有的只要import router,只依赖router这一个类,router里再去import 其他的ViewController,这样,就达到我们说的解耦。
一个基本完善的router,我认为应该有下面几个核心功能:
-
跳转ViewController
-
跳转服务
-
回传值
跳转ViewController是基本功能,这里包括跳转的时候,传入参数。
对于跳转,我们需要做到通过一个字符串,来跳转到我们想要的页面,那么我们首先要做的是将字符串和对应的页面关联起来,到时候,你给我这个字符串,我就知道你需要去哪个页面。
在URLNavigator里有一个注册方法,就是将字符串和需要跳转的ViewController关联起来。
navigator.register("navigator://user") { url, values, context in return UserViewController() }
更进一步,可以在字符串里把需要传递的参数也带上
navigator.register("navigator://user/") { url, values, context in guard let username = values["username"] as? String else { return nil } return UserViewController(navigator: navigator, username: username) }
这样不仅可以跳转到关联页面,还能传递参数。
register里面做的事很简单,就是用一个字典将字符串和和ViewController关联起来。
public typealias URLPattern = Stringprivate var viewControllerFactories = [URLPattern: ViewControllerFactory]()open func register(_ pattern: URLPattern, _ factory: @escaping ViewControllerFactory) { self.viewControllerFactories[pattern] = factory }
将字符串作为字典的key,创建ViewController的闭包作为value,就这样关联了字符串和ViewController。
在调用的时候,再根据字符串找到相应的闭包,得到ViewController,执行跳转动作。
open func viewController(for url: URLConvertible, context: Any? = nil) -> UIViewController? { let urlPatterns = Array(self.viewControllerFactories.keys) guard let match = self.matcher.match(url, from: urlPatterns) else { return nil } guard let factory = self.viewControllerFactories[match.pattern] else { return nil } return factory(url, match.values, context) }
从viewControllerFactories字典里拿到factory,再执行factory(url, match.values, context)。
这里的guard let match = self.matcher.match(url, from: urlPatterns) else { return nil }是拿到url里的参数,这里面参数的传入有一个自己定义的规则。URLMatcher.swift就是专门处理字符串的拆分,拿到参数。
然后
navigator.push("navigator://user/zhangsan")navigator.present("navigator://user/zhangsan")
调用服务,有时候,我们并不想跳到一个页面,仅仅是想调用某个类里面的某个函数。
navigator.register("navigator://user/") { url, values, context in guard let username = values["username"] as? String else { return nil } return UserViewController(navigator: navigator, username: username) }
这个是上面的注册代码,我们只要修改一下就可以了,调用服务,其他的操作一样
navigator.register("navigator://user/") { url, values, context in guard let username = values["username"] as? String else { return nil } //获取UserViewController对象 userVC,调用方法 userVC.callFuc(username: username) }
这个是我假想的一个方法。实际有一个和register类似的方法
private var handlerFactories = [URLPattern: URLOpenHandlerFactory]()open func handle(_ pattern: URLPattern, _ factory: @escaping URLOpenHandlerFactory) { self.handlerFactories[pattern] = factory }
有一个保存闭包和字符串对应关系的字典handlerFactories,不和register字典共用。
所以调用方法是
navigator.handle("navigator://user/") { (url, values, context) -> Bool in guard let username = values["username"] as? String else { return nil } //获取UserViewController对象 userVC,调用方法 userVC.callFuc(username: username) return true }
回传值,有时候,我们跳转到某个页面,需要这个页面执行后,把相关结果返回。
目前,URLNavigator还没有第三个功能。
总结一下:自己定义一个字符串规则,包含页面信息和参数信息,然后将字符串和对应的闭包关联起来,闭包可以是创建相应ViewController的操作,也可以是调用函数的操作,也可以是其他操作。在router通过字符串跳转的时候,拿到字符串,解析出参数,找到相应的闭包,将参数传给闭包执行,执行闭包得到的ViewController,拿去跳转。
作者:桔子听
作为一个开发者,有一个学习的氛围跟一个交流圈子特别重要这是一个我的iOS交流群:869685378,不管你是小白还是大牛欢迎入驻 ,分享BAT,阿里面试题、面试经验,讨论技术, 大家一起交流学习成长!