1. 程式人生 > >iOS-MVVM-框架介紹

iOS-MVVM-框架介紹

所以,MVVM 到底是什麼?與其專注於說明 MVVM 的來歷,不如讓我們看一個典型的 iOS 是如何構建的,並從那裡瞭解 MVVM:

我們看到的是一個典型的 MVC 設定。Model 呈現資料,View 呈現使用者介面,而 View Controller 調節它兩者之間的互動。Cool!

稍微考慮一下,雖然 View 和 View Controller 是技術上不同的元件,但它們幾乎總是手牽手在一起,成對的。你什麼時候看到一個 View 能夠與不同 View Controller 配對?或者反過來?所以,為什麼不正規化它們的連線呢?

這更準確地描述了你可能已經編寫的 MVC 程式碼。但它並沒有做太多事情來解決 iOS 應用中日益增長的重量級檢視控制器的問題。在典型的 MVC 應用裡,許多邏輯被放在 View Controller 裡。它們中的一些確實屬於 View Controller,但更多的是所謂的“表示邏輯(presentation logic)”,以 MVVM 屬術語來說,就是那些將 Model 資料轉換為 View 可以呈現的東西的事情,例如將一個 NSDate 轉換為一個格式化過的 NSString。

我們的圖解裡缺少某些東西,那些使我們可以把所有表示邏輯放進去的東西。我們打算將其稱為 “View Model” —— 它位於 View/Controller 與 Model 之間:

看起好多了!這個圖解準確地描述了什麼是 MVVM:一個 MVC 的增強版,我們正式連線了檢視和控制器,並將表示邏輯從 Controller 移出放到一個新的物件裡,即 View Model。MVVM 聽起來很複雜,但它本質上就是一個精心優化的 MVC 架構,而 MVC 你早已熟悉。

現在我們知道了什麼是 MVVM,但為什麼我們會想要去使用它呢?在 iOS 上使用 MVVM 的動機,對我來說,無論如何,就是它能減少 View Controller 的複雜性並使得表示邏輯更易於測試。通過一些例子,我們將看到它如何達到這些目標。

此處有三個重點是我希望你看完本文能帶走的:

MVVM 可以相容你當下使用的 MVC 架構。
MVVM 增加你的應用的可測試性。
MVVM 配合一個繫結機制效果最好。
如我們之前所見,MVVM 基本上就是 MVC 的改進版,所以很容易就能看到它如何被整合到現有使用典型 MVC 架構的應用中。讓我們看一個簡單的 Person Model 以及相應的 View Controller:

如我們之前所見,MVVM 基本上就是 MVC 的改進版,所以很容易就能看到它如何被整合到現有使用典型 MVC 架構的應用中。讓我們看一個簡單的 Person Model 以及相應的 View Controller:

  1. @interface Person : NSObject 
  2. - (instancetype)initwithSalutation:(NSString *)salutation firstName:(NSString *)firstName lastName:(NSString *)lastName birthdate:(NSDate *)birthdate; 
  3. @property (nonatomic, readonly) NSString *salutation; 
  4. @property (nonatomic, readonly) NSString *firstName; 
  5. @property (nonatomic, readonly) NSString *lastName; 
  6. @property (nonatomic, readonly) NSDate *birthdate; 
  7. @end

Cool!現在我們假設我們有一個 PersonViewController ,在 viewDidLoad 裡,只需要基於它的 model 屬性設定一些 Label 即可。

  1. - (void)viewDidLoad { 
  2.     [super viewDidLoad]; 
  3.     if (self.model.salutation.length > 0) { 
  4.         self.nameLabel.text = [NSString stringWithFormat:@"%@ %@ %@", self.model.salutation, self.model.firstName, self.model.lastName]; 
  5.     } else { 
  6.         self.nameLabel.text = [NSString stringWithFormat:@"%@ %@", self.model.firstName, self.model.lastName]; 
  7.     } 
  8.     NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init]; 
  9.     [dateFormatter setDateFormat:@"EEEE MMMM d, yyyy"]; 
  10.     self.birthdateLabel.text = [dateFormatter stringFromDate:model.birthdate]; 

這全都直截了當,標準的 MVC。現在來看看我們如何用一個 View Model 來增強它。

  1. @interface PersonViewModel : NSObject 
  2. - (instancetype)initWithPerson:(Person *)person; 
  3. @property (nonatomic, readonly) Person *person; 
  4. @property (nonatomic, readonly) NSString *nameText; 
  5. @property (nonatomic, readonly) NSString *birthdateText; 
  6. @end

我們的 View Model 的實現大概如下:

  1. @implementation PersonViewModel 
  2. - (instancetype)initWithPerson:(Person *)person { 
  3.     self = [super init]; 
  4.     if (!self) return nil; 
  5.     _person = person; 
  6.     if (person.salutation.length > 0) { 
  7.         _nameText = [NSString stringWithFormat:@"%@ %@ %@", self.person.salutation, self.person.firstName, self.person.lastName]; 
  8.     } else { 
  9.         _nameText = [NSString stringWithFormat:@"%@ %@", self.person.firstName, self.person.lastName]; 
  10.     } 
  11.     NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init]; 
  12.     [dateFormatter setDateFormat:@"EEEE MMMM d, yyyy"]; 
  13.     _birthdateText = [dateFormatter stringFromDate:person.birthdate]; 
  14.     return self; 
  15. @end

Cool!我們已經將 viewDidLoad 中的表示邏輯放入我們的 View Model 裡了。此時,我們新的 viewDidLoad 就會非常輕量:

- (void)viewDidLoad {
[super viewDidLoad];

self.nameLabel.text = self.viewModel.nameText;
self.birthdateLabel.text = self.viewModel.birthdateText;
}

所以,如你所見,並沒有對我們的 MVC 架構做太多改變。還是同樣的程式碼,只不過移動了位置。它與 MVC 相容,帶來更輕量的 View Controllers。

可測試,嗯?是怎樣?好吧,View Controller 是出了名的難以測試,因為它們做了太多事情。在 MVVM 裡,我們試著儘可能多的將程式碼移入 View Model 裡。測試 View Controller 就變得容易多了,因為它們不再做一大堆事情,並且 View Model 也非常易於測試。讓我們來看看:

  1. SpecBegin(Person) 
  2.     NSString *salutation = @"Dr."; 
  3.     NSString *firstName = @"first"; 
  4.     NSString *lastName = @"last"; 
  5.     NSDate *birthdate = [NSDate dateWithTimeIntervalSince1970:0]; 
  6.     it (@"should use the salutation available. ", ^{ 
  7.         Person *person = [[Person alloc] initWithSalutation:salutation firstName:firstName lastName:lastName birthdate:birthdate]; 
  8.         PersonViewModel *viewModel = [[PersonViewModel alloc] initWithPerson:person]; 
  9.         expect(viewModel.nameText).to.equal(@"Dr. first last"); 
  10.     }); 
  11.     it (@"should not use an unavailable salutation. ", ^{ 
  12.         Person *person = [[Person alloc] initWithSalutation:nil firstName:firstName lastName:lastName birthdate:birthdate]; 
  13.         PersonViewModel *viewModel = [[PersonViewModel alloc] initWithPerson:person]; 
  14.         expect(viewModel.nameText).to.equal(@"first last"); 
  15.     }); 
  16.     it (@"should use the correct date format. ", ^{ 
  17.         Person *person = [[Person alloc] initWithSalutation:nil firstName:firstName lastName:lastName birthdate:birthdate]; 
  18.         PersonViewModel *viewModel = [[PersonViewModel alloc] initWithPerson:person]; 
  19.         expect(viewModel.birthdateText).to.equal(@"Thursday January 1, 1970"); 
  20.     }); 
  21. SpecEnd 

如果我們沒有將這個邏輯移入 View Model,我們將不得不例項化一個完整的 View Controller 以及伴隨的 View,然後去比較我們 View 中 Lable 的值。這樣做不只是會變成一個麻煩的間接層,而且它只代表了一個十分脆弱的測試。現在,我們可以按意願自由地修改檢視層級而不必擔心破壞我們的單元測試。使用 MVVM 帶來的對於測試的好處非常清晰,甚至從這個簡單的例子來看也可見一斑,而在有更復雜的表示邏輯的情況下,這個好處會更加明顯。

注意到在這個簡單的例子中, Model 是不可變的,所以我們可以只在初始化的時候指定我們 View Model 的屬性。對於可變 Model,我們還需要使用一些繫結機制,這樣 View Model 就能在背後的 Model 改變時更新自身的屬性。此外,一旦 View Model 上的 Model 發生改變,那 View 的屬性也需要更新。Model 的改變應該級聯向下通過 View Model 進入 View。

在 OS X 上,我們可以使用 Cocoa 繫結,但在 iOS 上我們並沒有這樣好的配置可用。我們想到了 KVO(Key-Value Observation),而且它確實做了很偉大的工作。然而,對於一個簡單的繫結都需要很大的樣板程式碼,更不用說有許多屬性需要綁定了。作為替代,我個人喜歡使用 ReactiveCocoa,但 MVVM 並未強制我們使用 ReactiveCocoa。MVVM 是一個偉大的典範,它自身獨立,只是在有一個良好的繫結框架時做得更好。

相關推薦

iOS-MVVM-框架介紹

所以,MVVM 到底是什麼?與其專注於說明 MVVM 的來歷,不如讓我們看一個典型的 iOS 是如何構建的,並從那裡瞭解 MVVM: 我們看到的是一個典型的 MVC 設定。Model 呈現資料,View 呈現使用者介面,而 View Controller 調節它兩者之間的互動。Cool! 稍微考慮

IOS面向協議的MVVM架構介紹(來著網絡的鏈接)

客戶端 orien ade protocol eal target cocoa mvvm ref https://academy.realm.io/cn/posts/doios-natasha-murashev-protocol-oriented-mvvm/ 面向協議的

IOS APNS訊息推送框架介紹(pushy)以及詳細使用方法

最近公司需要做IOS訊息推送的功能,我負責後臺推送,IOS端資料處理以及回撥我不負責,本篇文章主要介紹基於java的apns訊息推送,使用框架為pushy。 宣告:我先前也沒有接觸過這個IOS推送,自己研究了兩天,通過百度,對比各個框架的優缺點,最後選擇了這個框架,有說的不對的地方,還

android開發框架總結 (二)MVP與MVVM詳細介紹與對比,如何選擇適合的框架(乾貨!)

前言 本篇文章將非常“細緻”地總結分析MVP與MVVM這兩種框架對於架構的選擇做了比較多的分析,應該是乾貨滿滿,如果你對這兩者的使用與選擇上還有迷惑之處。真的希望你能認真看完。 如果你是非常有經驗的程式猿,那就當相互學習總結,如果有不同看法還望指教。當然,我也是非常想進步的。  

Swift教程17-淡化MVC,使用MVVM框架開發輕巧便於維護的iOS app

MVVM是微軟提出一種移動開發框架,旨在針對傳統的MVC框架,解決傳統的MVC框架的控制器的臃腫問題. M: Model模型,也就是資料模型;比如一條微博,對應的所有欄位合成一條微博整體,這個整體就是Model V: View檢視,只用來顯

iOS MVVM+RAC 從框架到實戰

// // LSCircleListViewModel.m // ZhongShui // // Created by 王隆帥 on 16/3/10. // Copyright © 2016年 王隆帥. All rights reserved. // #import "LSCircleListVie

iOS開發-常用第三方開源框架介紹(絕對夠你用了)

影象: 1.圖片瀏覽控制元件MWPhotoBrowser        實現了一個照片瀏覽器類似 iOS 自帶的相簿應用,可顯示來自手機的圖片或者是網路圖片,可自動從網路下載圖片並進行快取。可對圖片進行縮放等操作。       下載:https:

iOS開發之利用MVVM框架來優化專案結構。對Controller瘦身以及MVC向MVVM框架的遷移。

MVC開發模式 : 1. 蘋果官方一直推薦我們開發者使用MVC的開發模式,所以我們大部分人之前的專案都是用MVC來開發APP,這樣開發,肯定會發現一個超級大的弊端,viewcontroller裡邊有大量的業務邏輯與檢視操作邏輯,隨著專案的不斷的迭代,會充斥著大量的問題,我們

iOS應用千萬級架構:MVVM框架

業務模組內的MVC和MVVM架構 目前,唯品會中MVC和MVVM架構並存,後期會偏重於MVVM架構的使用。 MVC架構 Model:程式中要操縱的實際物件的抽象,為Controller提供經過抽象的業務資料,供Controller排程 View:檢視,負責介面的元素的展示 Controller:控制器,管

關於前端mvc或mvvm框架數據跟蹤變化實現dom雙向綁定的原理

取值 cti angular 兩個 不同 lar log span 實例 一:最早的框架如backbone,實現對數據的變化監測是通過設置數據模型api。   比如其model對象管理的是數據,而修改這些數據就是通過固定的方法(set)來觸發事件從而更新dom, <

JAVA開發Web Service幾種框架介紹

需求 驚人的 總線 cast pri web服務 希望 uil blank 在講Web Service開發服務時,需要介紹一個目前開發Web Service的幾個框架,分別為Axis,axis2,Xfire,CXF以及JWS(也就是前面所述的JAX-WS,這是Java

Python之Web框架介紹

楊文 python gateway 應用程序 服務器 第三方 所有的語言Web框架本質其實就是起一個socket服務端,監聽一個端口,然後運行起來Web框架包含兩部分,一部分是socket,另外一部分是業務的邏輯處理,根據請求的不同做不同的處理Python的Web框架分成了兩類,即包含

Java集合框架介紹。Java Collection Frameworks = JCF

ava image src work ges frame 介紹 lec 分享 Java集合框架 = Java Collection Frameworks = JCF 。 為了方便理解,我畫了一張思維腦圖。 Java集合框架介紹。Java Collection F

HTML表格及框架介紹

url .html 排列 fan 水平對齊 邊框 位置 空心圓 掌握 一、列表1、有序列表ol   <ol type="1默認/a/A/i/I">    <li></li>   </ol> 2、無序列表ul(實際應

[iOS]FFmpeg框架iOS平臺上的編譯和使用

_id hex ips selector 文件 str content pat tps 使用環境 Mac OS Yosemite 10.10.5 開發工具 Xcode 7.0 Terminal 需要的文件鏈接 gas-preprocessor yasm F

ios測試框架的理解

gin behavior pla 高速 導入 pod should 創建 運行 關於ios的測試 Cedar 、Specta 、Kiwi 、 XCTest Specta和Kiwi的差別就是Kiwi包括了Specta和OCmock以及Expeata全部的功能

android mvp高速開發框架介紹(dileber的簡單介紹

activity 數據 -c pos androi mod family 基於 ebe 今天我為大家介紹一款android mvp框架:dileber(https://github.com/dileber/dileber.git) 官方交流qq群:171443

iOS MVVM+RAC

其中 屬於 微軟 做了 util 原創 都對 system 。。 〇、更新 針對最後一小節中幹掉基類的做法(Runtime + category),已經有所實現,並按照自己的思路新寫了個列表實現,具體可查看對於iOS架構模式之爭的一些思考。 一、前言 很早之前就想寫寫

fitnesse框架介紹

ref 相對 gin 測試開發 分離 align eight auto enter fitnesse框架介紹 2017-09-29 目錄: 1 fitnesse是什麽?2 框架介紹3 與junit、testng比較,fitnesse教其他框架有什麽優勢 1 f

主流服務框架介紹

良好的 國美 ken 名詞 我們 同時 技術 protocol 硬件 引用文章:http://www.cnblogs.com/wooer/p/7648825.html 當迎來服務SOA時代,我們面臨要解決的問題會很多,比如:系統的復雜度上升、服務依賴關系、服務性能監控、全鏈