国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 系統(tǒng) > iOS > 正文

如何在iOS上使用MVVM進(jìn)行路由詳解

2020-07-26 02:22:00
字體:
供稿:網(wǎng)友

前言

我已經(jīng)在幾個(gè)項(xiàng)目中使用MVVM了一段時(shí)間,我真的很喜歡它的簡單性。特別是,如果你像許多人一樣從MVC遷移,你只需要在你的架構(gòu)中增加一層ViewModel。如果您發(fā)現(xiàn)太多層級(jí)造成的復(fù)雜,這確實(shí)使事情變得更容易。

這是一個(gè)良好的開端,但這種簡單并不總是好的。在MVVM中,您將業(yè)務(wù)邏輯移出視圖控制器(VC),然后意識(shí)到它仍然很胖。視圖模型(VM)現(xiàn)在具有業(yè)務(wù)邏輯,但是展示數(shù)據(jù)(格式化)或路由如何?他們?nèi)匀槐焕г赩C中,我們需要將它們移出。

#示例流程

假設(shè)我們正在實(shí)現(xiàn)登陸頁面,如下所示。

##路由列表:

  • Login > 主頁面
  • Sign Up > 注冊(cè)頁面
  • Forgot Password(?) > 忘記密碼頁面

這看起來像是一個(gè)簡單的頁面,可以使用帶有3個(gè)segues的故事板來實(shí)現(xiàn)。但請(qǐng)相信我,事實(shí)并非如此。例如,您通常會(huì)在登錄時(shí)打開主屏幕。但在這種情況下,用戶的密碼可能已過期,您需要實(shí)施重定向到更改密碼屏幕。所以登錄路線變成:

  • Login > 主頁面 或者 更改密碼頁面

這是故事板路由失敗的地方。它無法處理這種動(dòng)態(tài)情況。所以你通常做的是讓VC處理它:

func loginButtonTapped() { // Start network request... // Upon response: if viewModel.shouldChangePassword { performSegue(id: "ChangePasswordScreen", sender: nil) } else { performSegue(id: "HomeScreen", sender: nil) }}

這是路由邏輯,它不應(yīng)該在VC中。如果您想要輕型VC,請(qǐng)?jiān)诰帉慽f語句之前三思而后行。他們是決定代碼,他們不屬于那里。根據(jù)我的理解,VC應(yīng)該只有視圖相關(guān)和粘合代碼。從來沒有決定代碼。
讓我們定義一個(gè)路由器協(xié)議,并從VC中取出這些if語句。我們會(huì)需要:

  • 路由ID:像segue ID一樣的一個(gè)字符串
  • 上下文:當(dāng)前視圖控制器是從哪里跳過來的
  • 可選的參數(shù):過渡所需的臨時(shí)數(shù)據(jù)。 (tableview點(diǎn)擊了哪一行等等)
protocol Router { func route( to routeID: String,  from context: UIViewController,  parameters: Any? )}

VC應(yīng)該只定義路由名稱,而不關(guān)心該路由的位置。這將是路由器的工作。

class LoginViewController: UIViewController {  enum Route: String {  case login  case signUp  case forgotPassword }  var viewModel: LoginViewModel! var router: Router!  ...  func loginButtonTapped() {  router.route(to: Route.login.rawValue, from: self) }  func signUpTapped() {  router.route(to: Route.signUp.rawValue, from: self) }  func forgotPasswordTapped() {  router.route(to: Route.forgotPassword.rawValue, from: self) }}

如上所述,登錄按鈕可以進(jìn)入主頁面或更改密碼頁面。那么路由器如何選擇正確的目的地呢?在這種情況下,您的路由器可能需要訪問您的VM。這樣,它可以直接讀取業(yè)務(wù)決策并決定目的地。

請(qǐng)注意VC已經(jīng)retain了VM和路由器。因此,路由器對(duì)VM應(yīng)該是weak/unowned引用。

class LoginRouter: Router {  unowned var viewModel: LoginViewModel  init(viewModel: LoginViewModel) {  self.viewModel = viewModel }  func route(  to routeID: String,   from context: UIViewController,   parameters: Any?)  {  guard let route = LoginVC.Route(rawValue: routeID) else {   return  }  switch route {  case .login:   if viewModel.shouldChangePassword {   // Push change-password-screen.   } else {   // Push home-screen.   }  case .signUp:   // Push sign-up-screen:   let vc = SignUpViewController()   let vm = SignUpViewModel()   vc.viewModel = vm   vc.router = SignUpRouter(viewModel: vm)   context.navigationController.push(vc, animated: true)  case . forgotPasswordScreen:   // Push forgot-password-screen.  } }}

總結(jié)

  • 我們完全將路由代碼移出VC。這有利于分離關(guān)注點(diǎn)。如果路由邏輯發(fā)生變化,您只需編輯路由器,而不是在VC中搜索push / present語句。
  • 隨著時(shí)間的推移,您將需要進(jìn)行許多設(shè)計(jì)更改。因此,保持視圖控制器輕量化是很重要的,因?yàn)樗c視圖緊密耦合的。在進(jìn)行UI大修時(shí),您不希望破壞路由邏輯。
  • 你不能用這種方法來使用故事板segue。我不知道我是否傷了你的心,但你不能用segues實(shí)現(xiàn)這樣的動(dòng)態(tài)流程。故事板應(yīng)該只對(duì)布局負(fù)責(zé)(同樣,關(guān)注點(diǎn)分離)

示例代碼:Movies (本地下載)

謝謝你的閱讀!

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對(duì)武林網(wǎng)的支持。

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 涞源县| 康定县| 海丰县| 临猗县| 滦平县| 临城县| 和硕县| 京山县| 宁国市| 白朗县| 宁阳县| 海淀区| 临泉县| 都匀市| 岳普湖县| 梅河口市| 湘潭县| 太保市| 琼中| 昂仁县| 临西县| 阳朔县| 平舆县| 天津市| 永春县| 新晃| 越西县| 阿拉善盟| 剑川县| 新绛县| 鹤壁市| 贡觉县| 三原县| 普兰店市| 电白县| 水富县| 新宁县| 和顺县| 华容县| 双鸭山市| 嘉兴市|