成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

iOS-MVVM-框架介紹

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

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

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

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

我會(huì)修改一個(gè)關(guān)鍵架構(gòu),并將其帶入我從那時(shí)起就在開發(fā)的各種應(yīng)用,即使用一種叫做 Model-View-ViewModel 的架構(gòu)替換 Model-View-Controller。

所以,MVVM 到底是什么?與其專注于說(shuō)明 MVVM 的來(lái)歷,不如讓我們看一個(gè)典型的 iOS 是如何構(gòu)建的,并從那里了解 MVVM:

 

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

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

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

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

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

現(xiàn)在我們知道了什么是 MVVM,但為什么我們會(huì)想要去使用它呢?在 iOS 上使用 MVVM 的動(dòng)機(jī),對(duì)我來(lái)說(shuō),無(wú)論如何,就是它能減少 View Controller 的復(fù)雜性并使得表示邏輯更易于測(cè)試。通過(guò)一些例子,我們將看到它如何達(dá)到這些目標(biāo)。

此處有三個(gè)重點(diǎn)是我希望你看完本文能帶走的:

MVVM 可以兼容你當(dāng)下使用的 MVC 架構(gòu)。
MVVM 增加你的應(yīng)用的可測(cè)試性。
MVVM 配合一個(gè)綁定機(jī)制效果最好。
如我們之前所見,MVVM 基本上就是 MVC 的改進(jìn)版,所以很容易就能看到它如何被整合到現(xiàn)有使用典型 MVC 架構(gòu)的應(yīng)用中。讓我們看一個(gè)簡(jiǎn)單的 Person Model 以及相應(yīng)的 View Controller:

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

現(xiàn)在我們知道了什么是 MVVM,但為什么我們會(huì)想要去使用它呢?在 iOS 上使用 MVVM 的動(dòng)機(jī),對(duì)我來(lái)說(shuō),無(wú)論如何,就是它能減少 View Controller 的復(fù)雜性并使得表示邏輯更易于測(cè)試。通過(guò)一些例子,我們將看到它如何達(dá)到這些目標(biāo)。

此處有三個(gè)重點(diǎn)是我希望你看完本文能帶走的:

MVVM 可以兼容你當(dāng)下使用的 MVC 架構(gòu)。
MVVM 增加你的應(yīng)用的可測(cè)試性。
MVVM 配合一個(gè)綁定機(jī)制效果最好。
如我們之前所見,MVVM 基本上就是 MVC 的改進(jìn)版,所以很容易就能看到它如何被整合到現(xiàn)有使用典型 MVC 架構(gòu)的應(yīng)用中。讓我們看一個(gè)簡(jiǎn)單的 Person Model 以及相應(yīng)的 View Controller:

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

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

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

這全都直截了當(dāng),標(biāo)準(zhǔn)的 MVC。現(xiàn)在來(lái)看看我們?nèi)绾斡靡粋€(gè) View Model 來(lái)增強(qiáng)它。

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

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

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

Cool!我們已經(jīng)將 viewDidLoad 中的表示邏輯放入我們的 View Model 里了。此時(shí),我們新的 viewDidLoad 就會(huì)非常輕量:

- (void)viewDidLoad {
[super viewDidLoad];

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

所以,如你所見,并沒有對(duì)我們的 MVC 架構(gòu)做太多改變。還是同樣的代碼,只不過(guò)移動(dòng)了位置。它與 MVC 兼容,帶來(lái)更輕量的 View Controllers。

可測(cè)試,嗯?是怎樣?好吧,View Controller 是出了名的難以測(cè)試,因?yàn)樗鼈冏隽颂嗍虑?。?MVVM 里,我們?cè)囍M可能多的將代碼移入 View Model 里。測(cè)試 View Controller 就變得容易多了,因?yàn)樗鼈儾辉僮鲆淮蠖咽虑?,并?View Model 也非常易于測(cè)試。讓我們來(lái)看看:

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

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

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

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

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

責(zé)任編輯:chenqingxiang 來(lái)源: 博客園
相關(guān)推薦

2017-02-24 10:02:04

AndroidMVVM應(yīng)用框架

2016-11-30 17:28:02

移動(dòng)開發(fā)iOSAndroid

2017-02-27 09:36:01

AndroidMVVM架構(gòu)

2013-06-20 10:28:39

MVVM框架avalon架構(gòu)

2012-05-11 09:38:00

MVVM Light

2019-01-29 10:06:31

開源技術(shù) 趨勢(shì)

2017-07-18 15:13:17

2009-06-19 14:49:43

Spring框架

2022-11-25 07:59:02

SpringJava

2009-08-18 09:17:03

AJAX框架

2023-05-29 09:18:28

.NET網(wǎng)絡(luò)通信

2009-07-15 10:23:51

Swing SWT集成

2016-03-30 09:34:27

2016-03-01 13:48:36

MVVMios快速開發(fā)

2022-02-16 09:01:13

iOSS開發(fā)XCode

2015-01-27 15:25:53

MVVM

2016-03-18 09:36:13

ios基礎(chǔ)框架

2012-06-01 11:10:07

iOS基本框架圖示

2009-12-15 18:39:36

Ruby Active

2010-03-02 16:09:53

WCF框架
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 一区二区高清不卡 | 拍拍无遮挡人做人爱视频免费观看 | 亚洲网站在线播放 | 色视频在线播放 | 91视频一区二区 | 91视频播放 | 日韩中文在线观看 | 亚洲精品一区二区网址 | 在线观看国产视频 | 国产三级国产精品 | 久久毛片| 成人免费高清 | 老司机午夜性大片 | 夜夜草 | 麻豆a级片 | 午夜影院在线观看视频 | 成人免费在线观看 | 日韩在线观看网站 | 欧美精品一二区 | 久久久久资源 | 亚洲成人中文字幕 | 欧美精品一级 | 激情影院久久 | 精品免费国产一区二区三区 | 亚洲国产精品福利 | 欧美激情久久久 | 一道本不卡 | 国产精品69av| 国产成人精品免高潮在线观看 | 欧美区在线 | 中文字幕一区二区三区精彩视频 | 亚洲精品久久久久久一区二区 | 综合色站导航 | 精品一区二区三区av | 国产在线成人 | 久久久精彩视频 | 成人不卡 | 亚洲成人一二区 | 在线播放第一页 | 精品一区二区三区在线观看国产 | 一级片网址 |