在线观看不卡亚洲电影_亚洲妓女99综合网_91青青青亚洲娱乐在线观看_日韩无码高清综合久久

鍍金池/ 教程/ iOS/ MVVM 介紹
與四軸無人機的通訊
在沙盒中編寫腳本
結構體和值類型
深入理解 CocoaPods
UICollectionView + UIKit 力學
NSString 與 Unicode
代碼簽名探析
測試
架構
第二期-并發(fā)編程
Metal
自定義控件
iOS 中的行為
行為驅(qū)動開發(fā)
Collection View 動畫
截圖測試
MVVM 介紹
使 Mac 應用數(shù)據(jù)腳本化
一個完整的 Core Data 應用
插件
字符串
為 iOS 建立 Travis CI
先進的自動布局工具箱
動畫
為 iOS 7 重新設計 App
XPC
從 NSURLConnection 到 NSURLSession
Core Data 網(wǎng)絡應用實例
GPU 加速下的圖像處理
自定義 Core Data 遷移
子類
與調(diào)試器共舞 - LLDB 的華爾茲
圖片格式
并發(fā)編程:API 及挑戰(zhàn)
IP,TCP 和 HTTP
動畫解釋
響應式 Android 應用
初識 TextKit
客戶端
View-Layer 協(xié)作
回到 Mac
Android
Core Image 介紹
自定義 Formatters
Scene Kit
調(diào)試
項目介紹
Swift 的強大之處
測試并發(fā)程序
Android 通知中心
調(diào)試:案例學習
從 UIKit 到 AppKit
iOS 7 : 隱藏技巧和變通之道
安全
底層并發(fā) API
消息傳遞機制
更輕量的 View Controllers
用 SQLite 和 FMDB 替代 Core Data
字符串解析
終身學習的一代人
視頻
Playground 快速原型制作
Omni 內(nèi)部
同步數(shù)據(jù)
設計優(yōu)雅的移動游戲
繪制像素到屏幕上
相機與照片
音頻 API 一覽
交互式動畫
常見的后臺實踐
糟糕的測試
避免濫用單例
數(shù)據(jù)模型和模型對象
Core Data
字符串本地化
View Controller 轉場
照片框架
響應式視圖
Square Register 中的擴張
DTrace
基礎集合類
視頻工具箱和硬件加速
字符串渲染
讓東西變得不那么糟
游戲中的多點互聯(lián)
iCloud 和 Core Data
Views
虛擬音域 - 聲音設計的藝術
導航應用
線程安全類的設計
置換測試: Mock, Stub 和其他
Build 工具
KVC 和 KVO
Core Image 和視頻
Android Intents
在 iOS 上捕獲視頻
四軸無人機項目
Mach-O 可執(zhí)行文件
UI 測試
值對象
活動追蹤
依賴注入
Swift
項目管理
整潔的 Table View 代碼
Swift 方法的多面性
為什么今天安全仍然重要
Core Data 概述
Foundation
Swift 的函數(shù)式 API
iOS 7 的多任務
自定義 Collection View 布局
測試 View Controllers
訪談
收據(jù)驗證
數(shù)據(jù)同步
自定義 ViewController 容器轉場
游戲
調(diào)試核對清單
View Controller 容器
學無止境
XCTest 測試實戰(zhàn)
iOS 7
Layer 中自定義屬性的動畫
第一期-更輕量的 View Controllers
精通 iCloud 文檔存儲
代碼審查的藝術:Dropbox 的故事
GPU 加速下的圖像視覺
Artsy
照片擴展
理解 Scroll Views
使用 VIPER 構建 iOS 應用
Android 中的 SQLite 數(shù)據(jù)庫支持
Fetch 請求
導入大數(shù)據(jù)集
iOS 開發(fā)者的 Android 第一課
iOS 上的相機捕捉
語言標簽
同步案例學習
依賴注入和注解,為什么 Java 比你想象的要好
編譯器
基于 OpenCV 的人臉識別
玩轉字符串
相機工作原理
Build 過程

MVVM 介紹

我于 2011 年在 500px 找到自己的第一份 iOS 開發(fā)工作。雖然我已經(jīng)在大學里做了好幾年 iOS 外包開發(fā),但這才是我的一個真正的 iOS 開發(fā)工作。我被作為唯一的 iOS 開發(fā)者被招聘去實現(xiàn)擁有漂亮設計的 iPad 應用。在短短七周里,我們就發(fā)布了 1.0 并持續(xù)迭代,添加了更多特性,但從本質(zhì)上,代碼庫也變得更加復雜了。

有時我感覺就像我不知道在做什么。雖然我知道自己的設計模式——就像任何好的編程人員那樣 —— 但我太接近我在做的產(chǎn)品以至于不能客觀地衡量我的架構決策的有效性。當隊伍中來了另外一位開發(fā)者時,我意識到我們陷入困境了。

從沒聽過 MVC ?有人稱之為 Massive View Controller(重量級視圖控制器),這就是我們那時候的感覺。我不打算介紹令人汗顏的細節(jié),但說實在的,如果我不得不再次重來一次,我絕對會做出不同的決策。

我會修改一個關鍵架構,并將其帶入我從那時起就在開發(fā)的各種應用,即使用一種叫做 Model-View-ViewModel 的架構替換 Model-View-Controller。

所以,MVVM 到底是什么?與其專注于說明 MVVM 的來歷,不如讓我們看一個典型的 iOS 是如何構建的,并從那里了解 MVVM:

http://wiki.jikexueyuan.com/project/objc/images/13-1.png" alt="" />

我們看到的是一個典型的 MVC 設置。Model 呈現(xiàn)數(shù)據(jù),View 呈現(xiàn)用戶界面,而 View Controller 調(diào)節(jié)它兩者之間的交互。Cool!

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

http://wiki.jikexueyuan.com/project/objc/images/13-2.png" alt="" />

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

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

http://wiki.jikexueyuan.com/project/objc/images/13-3.png" alt="" />

看起好多了!這個圖解準確地描述了什么是 MVVM:一個 MVC 的增強版,我們正式連接了視圖和控制器,并將表示邏輯從 Controller 移出放到一個新的對象里,即 View Model。MVVM 聽起來很復雜,但它本質(zhì)上就是一個精心優(yōu)化的 MVC 架構,而 MVC 你早已熟悉。

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

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

  • MVVM 可以兼容你當下使用的 MVC 架構。
  • MVVM 增加你的應用的可測試性。
  • MVVM 配合一個綁定機制效果最好。

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

@interface Person : NSObject

- (instancetype)initwithSalutation:(NSString *)salutation firstName:(NSString *)firstName lastName:(NSString *)lastName birthdate:(NSDate *)birthdate;

@property (nonatomic, readonly) NSString *salutation;
@property (nonatomic, readonly) NSString *firstName;
@property (nonatomic, readonly) NSString *lastName;
@property (nonatomic, readonly) NSDate *birthdate;

@end

Cool!現(xiàn)在我們假設我們有一個 PersonViewController ,在 viewDidLoad 里,只需要基于它的 model 屬性設置一些 Label 即可。

- (void)viewDidLoad {
    [super viewDidLoad];

    if (self.model.salutation.length > 0) {
        self.nameLabel.text = [NSString stringWithFormat:@"%@ %@ %@", self.model.salutation, self.model.firstName, self.model.lastName];
    } else {
        self.nameLabel.text = [NSString stringWithFormat:@"%@ %@", self.model.firstName, self.model.lastName];
    }

    NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
    [dateFormatter setDateFormat:@"EEEE MMMM d, yyyy"];
    self.birthdateLabel.text = [dateFormatter stringFromDate:model.birthdate];
}

這全都直截了當,標準的 MVC?,F(xiàn)在來看看我們?nèi)绾斡靡粋€ View Model 來增強它。

@interface PersonViewModel : NSObject

- (instancetype)initWithPerson:(Person *)person;

@property (nonatomic, readonly) Person *person;

@property (nonatomic, readonly) NSString *nameText;
@property (nonatomic, readonly) NSString *birthdateText;

@end

我們的 View Model 的實現(xiàn)大概如下:

@implementation PersonViewModel

- (instancetype)initWithPerson:(Person *)person {
    self = [super init];
    if (!self) return nil;

    _person = person;
    if (person.salutation.length > 0) {
        _nameText = [NSString stringWithFormat:@"%@ %@ %@", self.person.salutation, self.person.firstName, self.person.lastName];
    } else {
        _nameText = [NSString stringWithFormat:@"%@ %@", self.person.firstName, self.person.lastName];
    }

    NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
    [dateFormatter setDateFormat:@"EEEE MMMM d, yyyy"];
    _birthdateText = [dateFormatter stringFromDate:person.birthdate];

    return self;
}

@end

Cool!我們已經(jīng)將 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 也非常易于測試。讓我們來看看:

SpecBegin(Person)
    NSString *salutation = @"Dr.";
    NSString *firstName = @"first";
    NSString *lastName = @"last";
    NSDate *birthdate = [NSDate dateWithTimeIntervalSince1970:0];

    it (@"should use the salutation available. ", ^{
        Person *person = [[Person alloc] initWithSalutation:salutation firstName:firstName lastName:lastName birthdate:birthdate];
        PersonViewModel *viewModel = [[PersonViewModel alloc] initWithPerson:person];
        expect(viewModel.nameText).to.equal(@"Dr. first last");
    });

    it (@"should not use an unavailable salutation. ", ^{
        Person *person = [[Person alloc] initWithSalutation:nil firstName:firstName lastName:lastName birthdate:birthdate];
        PersonViewModel *viewModel = [[PersonViewModel alloc] initWithPerson:person];
        expect(viewModel.nameText).to.equal(@"first last");
    });

    it (@"should use the correct date format. ", ^{
        Person *person = [[Person alloc] initWithSalutation:nil firstName:firstName lastName:lastName birthdate:birthdate];
        PersonViewModel *viewModel = [[PersonViewModel alloc] initWithPerson:person];
        expect(viewModel.birthdateText).to.equal(@"Thursday January 1, 1970");
    });
SpecEnd

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

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

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

我們覆蓋了不少內(nèi)容:從普通的 MVC 派生出 MVVM,看它們是如何相兼容的范式,從一個可測試的例子觀察 MVVM,并看到 MVVM 在有一個配對的綁定機制時工作得更好。如果你有興趣學習更多關于 MVVM 的知識,你可以看看這篇博客,它用更多細節(jié)解釋了 MVVM 的好處,或者這一篇關于我們?nèi)绾卧谧罱捻椖坷锸褂?MVVM 獲得巨大的成功的文章。我同樣還有一個經(jīng)過完整測試,基于 MVVM 的應用,叫做 C-41 ,它是開源的。去看看吧,如果你有任何疑問,請告訴我。