亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

[iOS]MVVM-框架介紹

2019-11-14 19:08:24
字體:
來源:轉載
供稿:網友
  

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

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

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

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

所以,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:

1
2
3
4
5
6
7
8
9
10
@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!現在我們假設我們有一個 PersonViewController ,在 viewDidLoad 里,只需要基于它的 model 屬性設置一些 Label 即可。

1
2
3
4
5
6
7
8
9
10
11
12
13
- (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在來看看我們如何用一個 View Model 來增強它。

1
2
3
4
5
6
7
8
9
10
@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 的實現大概如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
@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!我們已經將 viewDidLoad 中的表示邏輯放入我們的 View Model 里了。此時,我們新的 viewDidLoad 就會非常輕量:

1
2
3
4
5
6
- (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
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
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在,我們可以按意愿自由地修改視圖層級而不必擔心破壞我們的單元測試。使用 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 是一個偉大的典范,它自身獨立,只是在有一個良好的綁定框架時做得更好。

我們覆蓋了不少內容:從普通的 MVC 派生出 MVVM,看它們是如何相兼容的范式,從一個可測試的例子觀察 MVVM,并看到 MVVM 在有一個配對的綁定機制時工作得更好。如果你有興趣學習更多關于 MVVM 的知識,你可以看看這篇博客,它用更多細節解釋了 MVVM 的好處,或者這一篇關于我們如何在最近的項目里使用 MVVM 獲得巨大的成功的文章。我同樣還有一個經過完整測試,基于 MVVM 的應用,叫做 C-41 ,它是開源的。

 


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
成人性生交大片免费看小说| 中文字幕一区电影| 爱福利视频一区| 热久久视久久精品18亚洲精品| 日韩精品在线看| 色婷婷综合成人av| 欧美精品videossex性护士| 日本成人在线视频网址| 在线观看国产欧美| 精品中文字幕乱| 国产精品久久国产精品99gif| 国产成一区二区| 国产精品色午夜在线观看| 国产精品国模在线| 欧美一级在线播放| 91av视频在线免费观看| 亚洲国产欧美在线成人app| 国产日本欧美在线观看| 欧美另类69精品久久久久9999| 亚洲级视频在线观看免费1级| 九色精品美女在线| 91久久综合亚洲鲁鲁五月天| 国产精品视频26uuu| 欧美在线视频观看免费网站| 亚洲欧美日韩在线一区| 国产在线拍偷自揄拍精品| 久久精品中文字幕电影| 操人视频在线观看欧美| 亚洲国模精品一区| 57pao国产精品一区| 欧美性色视频在线| 午夜伦理精品一区| 4438全国亚洲精品在线观看视频| 成人羞羞国产免费| 久久久久久久久中文字幕| 精品久久中文字幕| 色久欧美在线视频观看| **欧美日韩vr在线| 日本精品久久久久久久| 色狠狠av一区二区三区香蕉蜜桃| 夜夜狂射影院欧美极品| 欧美黑人xxxⅹ高潮交| 成人免费淫片aa视频免费| 成人中文字幕+乱码+中文字幕| 国产精品91免费在线| 成人xxxxx| 不卡在线观看电视剧完整版| 亚洲人成在线观| 国产va免费精品高清在线| 欧美日韩中文字幕在线视频| 国产精品高清网站| 国产欧美精品xxxx另类| 日韩精品999| 日本人成精品视频在线| 欧美日韩免费观看中文| 亚洲国产精品嫩草影院久久| 国产精品96久久久久久又黄又硬| 国产精品美女在线观看| 日韩av高清不卡| 色综合五月天导航| 色综合视频一区中文字幕| 久久久久久久色| 欧美xxxx18国产| 欧美孕妇与黑人孕交| 成人看片人aa| 日本免费久久高清视频| 欧美成人精品xxx| 成人久久18免费网站图片| 亚洲国产天堂久久综合| 日韩成人av网址| 亚洲国产精品久久久久| 91极品视频在线| 一区二区欧美激情| 日韩免费观看高清| 亚洲a∨日韩av高清在线观看| 国产做受69高潮| 成人久久精品视频| 97视频在线观看免费高清完整版在线观看| 久久九九精品99国产精品| 亲爱的老师9免费观看全集电视剧| 欧美巨乳美女视频| 久久久久久久久久国产精品| 日韩成人在线观看| 久久免费观看视频| 日韩精品在线电影| 黄色一区二区在线观看| 国产成人综合一区二区三区| 狠狠躁夜夜躁久久躁别揉| 亚洲视频在线观看免费| 欧美在线一区二区视频| 色琪琪综合男人的天堂aⅴ视频| 另类少妇人与禽zozz0性伦| 国产精品久久久久久久久久久久久久| 黄色一区二区在线| 欧美日韩国产综合新一区| 91精品国产电影| 中日韩美女免费视频网站在线观看| 精品中文字幕久久久久久| 国产精品久久久久久久久久免费| 亚洲字幕在线观看| 成人国产亚洲精品a区天堂华泰| 久久久精品免费视频| 亚洲性无码av在线| 亚洲www视频| 久久精品国产欧美亚洲人人爽| 久久黄色av网站| 国产精品福利网| 久久视频精品在线| 久久99热这里只有精品国产| 日韩美女视频在线观看| 国产精品观看在线亚洲人成网| 亚洲成人aaa| 欧美在线亚洲在线| 国产精品偷伦视频免费观看国产| 91国内揄拍国内精品对白| 欧美日韩在线观看视频小说| 国产精品扒开腿做爽爽爽的视频| 自拍偷拍亚洲精品| 欧美日韩一区二区免费在线观看| 大伊人狠狠躁夜夜躁av一区| 7m精品福利视频导航| 91精品国产综合久久久久久久久| 国产精品一区二区三区免费视频| 日韩在线视频国产| 国产精品一区电影| 欧美午夜精品久久久久久人妖| 久久久久久九九九| 最近2019中文字幕mv免费看| 91麻豆桃色免费看| 欧美日韩亚洲视频| 日韩精品免费在线视频| 亚洲精品福利免费在线观看| 亚洲天堂视频在线观看| 日日狠狠久久偷偷四色综合免费| 精品久久久久久久久久久久久| 国产成人精品久久亚洲高清不卡| 91久久精品国产91久久性色| 黄色精品一区二区| 91香蕉嫩草神马影院在线观看| 国产成人avxxxxx在线看| 欧美中文在线观看国产| 中文字幕最新精品| 亚洲欧洲av一区二区| 精品国产乱码久久久久久天美| 国产日韩欧美夫妻视频在线观看| 成人亚洲激情网| 日韩色av导航| 亚洲激情在线观看| 欧美亚洲在线播放| 伊人伊成久久人综合网小说| 日韩大胆人体377p| 亚洲精品中文字幕有码专区| 正在播放欧美视频| 亚洲黄色片网站| 亚洲日本成人女熟在线观看| 国产精品一区二区三区久久久| 亚洲男人天堂网| 中文字幕国内精品| 日韩一区二区三区xxxx| 2018日韩中文字幕| 欧美噜噜久久久xxx| 亚洲成人在线视频播放| 久久久亚洲影院| 国产精品久久久久久亚洲影视|