深入解析設計模式中的裝飾器模式在iOS應用開發中的實現_IOS

來源:互聯網
上載者:User

裝飾器模式可以在不修改代碼的情況下靈活的為一對象添加行為和職責。當你要修改一個被其它類包含的類的行為時,它可以代替子類化方法。

一、基本實現
下面我把類的結構圖向大家展示如下:

讓我們簡單分析一下上面的結構圖,Component是定義一個對象介面,可以給這些對象動態地添加職責。ConcreteComponent是定義了一個具體的對象,也可以給這個對象添加一些職責。Decorator,裝飾抽象類別,繼承了Component,從外類來擴充Component類的功能,但對於Component來說,是無需知道Decorator的存在的。至於ConcreteDecorator就是具體的裝飾對象,起到給Component添加職責的功能。

下面,還是老套路,我會儘可能的給出Objective C實現的最簡單的執行個體代碼,首先聲明一下,這些代碼是運行在ARC環境下的,所以對於某些可能引起記憶體流失的資源並沒有採用手動釋放的方式,這一點還是需要大家注意。

注意:本文所有代碼均在ARC環境下編譯通過。

Components類介面檔案

複製代碼 代碼如下:

#import<Foundation/Foundation.h>    

@interface Components :NSObject
-(void) Operation;
@end


Components類實現檔案

#import"Components.h"

複製代碼 代碼如下:

@implementation Components
-(void)Operation{
    return;
}
@end

Decorator類介面檔案

#import"Components.h"  

複製代碼 代碼如下:

@interface Decorator :Components{
@protected Components *components;
}
-(void)SetComponents:(Components*)component;
@end

Decorator類實現檔案

#import"Decorator.h"

複製代碼 代碼如下:

@implementation Decorator
-(void)SetComponents:(Components*)component{
    components = component;
}
-(void)Operation{
    if(components!=nil){
        [components Operation];
    }
}
@end

ConcreteComponent類介面檔案
複製代碼 代碼如下:

#import"Components.h"

@interface ConcreteComponent :Components
@end


ConcreteComponent類實現檔案
複製代碼 代碼如下:

#import "ConcreteComponent.h"

@implementation ConcreteComponent
-(void)Operation{
    NSLog(@"具體操作的對象");
}
@end


ConcreteDecoratorA類介面檔案
複製代碼 代碼如下:

#import "ConcreteDecoratorA.h"
#import "Decorator.h"

@interface ConcreteDecoratorA :Decorator
@end


ConcreteDecoratorA類實現檔案

#import"ConcreteDecoratorA.h"

複製代碼 代碼如下:

@implementation ConcreteDecoratorA
-(void)Operation{
    NSLog(@"具體裝飾對象A的操作");
    [super Operation];
}
@end

ConcreteDecoratorB類介面檔案

#import "Decorator.h"

複製代碼 代碼如下:

@interface ConcreteDecoratorB :Decorator
@end

ConcreteDecoratorB類實現檔案
複製代碼 代碼如下:

#import "ConcreteDecoratorB.h"

@implementation ConcreteDecoratorB
-(void)Operation{
    NSLog(@"具體裝飾對象B的操作");
    [super Operation];
}
@end


Main方法
複製代碼 代碼如下:

#import <Foundation/Foundation.h>
#import "ConcreteComponent.h"
#import "ConcreteDecoratorA.h"
#import "ConcreteDecoratorB.h"

int main (int argc,const char* argv[])
{
    @autoreleasepool{
        ConcreteComponent *c = [[ConcreteComponent alloc]init];
        ConcreteDecoratorA *d1 = [[ConcreteDecoratorA alloc]init];
        ConcreteDecoratorB *d2 = [[ConcreteDecoratorB alloc]init];
        [d1 SetComponents:c];
        [d2 SetComponents:d1];
        [d2 Operation];
    }
    return 0;
}


好啦,上面是需要展示的類,文法上都很簡單,沒有什麼需要重點說的,可能值得一提的是關於子類調用父類方法的知識點,就是在調用每個對象的Operation方法的時候,裡面會有一句代碼是[super Operation];這句代碼構很關鍵,他構成了各個對象之間Operation方法的跳轉,以此完成對Components類對象的”裝飾”。

二、分類(Category)和委託(Delegation)
在 Object-C 裡有兩個種非常常見的實現模式:分類(Category)和委託(Delegation)。

1.分類 Category

分類是一種非常強大的機制,它允許你在一個已存在的類裡添加新方法,而不需要去為他添加一個子類。新方法在編譯的時候添加,它能像這個類的擴充方法一樣正常執行。一個裝飾器跟類的定義稍微有點不同的就是,因為裝飾器不能被執行個體化,它只是一個擴充。

提示:除了你自己類的擴充,你還可在任何 Cocoa 類裡的擴充添加方法。
如何使用分類:

現在你有一個 Album 對象,你需要把它顯示在一個表單視圖裡(table view):

專輯的標題從哪裡來?Album 只是一個模型對象,它才不會去關心你如果去顯示這些資料。為了這些,你需要給 Album 類添加一些額外的代碼,但是請不要直接修改這個類。

你現在就需要為 Album 添加一個分類 (category) 的擴充;它將定義一個新地方法用來返回一個資料結構,這個資料結構可以很容易的被 UITableViews 使用。

這個資料結構看起來如下:

為 Album 添加一個分類,導航 File\New\File… 選擇 Object-C category 模版─不要習慣的去選擇 Object-C class,在 Category 後面輸入 TableRepresentation,Category to 後面輸入 Album。

提示:你有沒有注意這個新檔案的名字?Album+TableRepresentation 說明它是 Album 類的一個擴充。這個習慣很重要,因為第一這很容易讀,第二防止你或者其他人建立的分類跟其衝突。
開啟 Album+TableRepresentation,加入下面的方法原型:

- (NSDictionary*)tr_tableRepresentation;
注意,這是一個 tr_ 開頭的方法名,就像是這個分類名字的縮寫一樣:TableRepresentation。其次,這個習慣會避免這個方法跟其它方法重名!

提示:如果分類 (Category) 聲明的一個方法跟原始類的一個方法重名,或者跟同類裡的的另一個分類名字重複(或者是它的父類),當它在啟動並執行時候,它就不知道要執行哪個方法。如果是在你自己類的分類裡,它不太可能出現大的問題,但是如果一個標準 Cocoa 或者 Cocoa Touch 類裡面添加這個分類的方法,就可能會引起嚴重的問題。
開啟 Album+TableRepersentation.m 檔案添加下面的方法:

複製代碼 代碼如下:

- (NSDictionary*)tr_TableRepersentation
{
    return @{@"titles":@[@"Artist", @"Album", @"Genre", @"Year"],
            @"values":@[self.artist, self.title, self.genre, self.year]};
};

考慮一會,為什麼這種模式如些強大:

你能夠直接使用 Album 的屬性。
你已經添加在 Album 類裡,但它並不是它的子類。如果子類需要,你同樣也可以這樣做。
這樣一個簡單的添加,Album 類的資料返回一個 UITableView 可用的資料結構,但並不需要修改 Album 的代碼。
蘋果在基礎類裡大量的使用了分類設計模式。去看看他們是怎麼做的,開啟 NSString.h。找到 @interface NSString,你將會看到這個類定義了三個分類:NSStringExtensionMethods, NSExtendedStringPropertyListParsing 和 NSStingDeprecated。在代碼片裡,分類將協助你保持方法的組織性和分離必。

2.委託 Delegation

另外一種裝飾器的設計模式是,委託 (Delegation),它是一種機制,一個對象代表另外一個對象或者其相互合作。例子,當你使用 UITableView 的時候,其中一個方法是你必需要執行的,tableView:numberOfRowsInSection:。

你可能並不期望 UITableView 知道每個 section 中有多少行,這是程式的特性。因此,計算每個 section 有多少行的工作就交給了 UITableView 的委託 (delegate)。它允許 UITableView 類不依賴它顯示的資料。

當你建立了一個新的 UITableView 的時候,這裡有一個類似的解釋:

UITableView 對象的工作就是顯示一個表單視圖。然而,最終它都需要一些它資訊,它並不擁有這些資訊。然後,它會轉向它的委託,發送一個添加資訊的訊息。在 Object-C 中實現委託模式,一個類可以通過協議 (protocol) 來聲明一個可選和必選的方法。稍後,在這個教程你將覆蓋一個協議 (protocols)。

它看起來比子類更容易,覆蓋需要的方法,但是考慮如果是單類的話你只能建立子類。如果你想一個對象委託兩個或者多個對象的時候,子類化的方法是不能實現的。

提示:這是一個很重要的模式。蘋果在 UIKit 類中大量的使用了此方法:UITableView, UITextView, UITextField, UIWebView, UIAlert, UIActionSheet, UICollectionView, UIPickerView, UIGestureRecognizer, UIScrollView。這個列表還可以有很多。
如何使用委託模式:

開啟 ViewController.m,在頂部引入如下檔案

複製代碼 代碼如下:

#import "LibraryAPI.h"
#import "Album+TableRepresentation.h"



現在,在類的擴充裡的添加一些私人變數,它們看起來如下:
複製代碼 代碼如下:

@interface ViewController (){
    UITableView *dataTable;
    NSArray *allAlbums;
    NSDictionary *currentAlbumData;
    int currentAlbumIndex;
}
@end

現在,替換類擴充裡的 @interface 這一行,完成後如下:
複製代碼 代碼如下:

@interface ViewController () <UITableViewDataSoure, UITableViewDelegate> {

這就是如何設定一個正確的委託─把它相象成允許一個委託來履行一個方法的合約。這裡,表明 ViewController 將會遵照 UITableViewDataSource 和 UITableViewDelegate 協議。這種方法下 UITableView 必須執行它自己的委託方法。

下面,用下面的代碼替換 viewDidLoad:

複製代碼 代碼如下:

- (void)viewDidLoad {
    [super viewDidLoad];
    // 1
    self.view.backgroundColor = [UIColor colorWithRed:0.76f green:0.81f blue:0.87f alpha:1];
    currentAlbumIndex = 0;

    //2
    allAlbums = [[LibraryAPI sharedInstance] getAlbums];

    // 3
    // the uitableview that presents the album data
    dataTable = [[UITableView alloc] initWithFrame:CGRectMake(0, 120, self.view.frame.size.width, self.view.frame.size.height-120) style:UITableViewStyleGrouped];
    dataTable.delegate = self;
    dataTable.dataSource = self;
    dataTable.backgroundView = nil;
    [self.view addSubview:dataTable];
}


這裡分析下上面的代碼:

把背景色改為漂亮的深藍色。
從 API 擷取一個列表,它包含所有的專輯資料。不能直接使用 PersistencyManager。
建立一個 UITableView。你聲明了視圖控制器是 UITableView delegate/data source;因此,UITableView 將會提供視圖控制器需要的所有資訊。
現在,在 ViewController.m 裡面添加如下方法:

複製代碼 代碼如下:

- (void)showDataForAlbumAtIndex:(int)albumIndex{
    // defensive code: make sure the requested index is lower than the amount of albums
    if (albumIndex < allAlbums.count) {
        // fetch the album
        Album *album = allAlbums[albumIndex];
        // save the albums data to present it later in the tableview
        currentAlbumData = [album tr_tableRepresentation];
    } else {
        currentAlbumData = nil;
    }

    // we have the data we need, let's refresh our tableview
    [dataTable reloaddata];
}


showDataForAlbumAtIndex: 從專輯數組中取出需要的專輯資料。當你需要顯示新資料的時候,你只需要重載資料 (relaodData)。這是因為 UITableView 需要請求它的委託代理,像有多少 sections 將會在表單視圖中顯示,每個 section 中有多少行,每行看起來是什麼樣的。

在 viewDidLoad 中添加下面代碼

複製代碼 代碼如下:

[self showDataForAlbumAtIndex:currentAlbumIndex];

當程式啟動並執行時候它會載入當前的專輯資訊。由於 currentAlbumIndex 的預設值為 0,所以會顯示收藏中的第一張專輯資訊。

構建並運行你的項目,你的程式會崩潰掉,在控制台會輸入如下的異常:

出現什麼問題了?你已經聲明了 ViewController 中的 UItableView 的委託(delegate)和資料來源(data source)。但是在這種情況下,你必需執行所有的必需方法─包含 tableView:numberOfRowsInsection:─你現在還沒有它。

在 ViewContrller.m 的 @implementation 和 @end 的任何地方添加如下代碼:

複製代碼 代碼如下:

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
    return [currentAlbumData[@"titles"] count];
}

- (UITableViewCell*)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"cell"];
    if (!cell) {
        cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleValue1 reuseIdentifier:@"cell"];
    }

    cell.textLabel.text = currentAlbumData[@"titles"][indexPath.row];
    cell.detailTextLabel.text = currentAlbumData[@"values"][indexPath.row];

    return cell;
}


tableView:numberOfRowsIndexSection: 返回表單視圖顯示的行數,匹配資料結構中標題的數目。

tableView:cellForRowAtIndexPath: 建立並返回一個帶標題和資訊的 cell。

現在構建並運行你的項目。你的程式開始運行並顯示出下圖的介面:

這目前為止事情看起來很不錯。但是如果你回過去看第一張圖片的時候,你會發現在螢幕的頂端有一個可以水平滾動的視圖,用於切換專輯。它只是簡單的水平滾動,為什麼不做一個可以重複使用的視圖來代替它呢。

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.