【原】談談對Objective-C中代理模式的誤解

来源:http://www.cnblogs.com/polobymulberry/archive/2016/06/04/5559075.html
-Advertisement-
Play Games

【原】談談對Objective-C中代理模式的誤解 本文轉載請註明出處 —— polobymulberry-博客園 1. 前言 這篇文章主要是對代理模式和委托模式進行了對比,個人認為Objective-C中的delegate大部分用法屬於委托模式。全文有些摳概念,對實際開發沒有任何影響。 前段時間看 ...


【原】談談對Objective-C中代理模式的誤解

本文轉載請註明出處 —— polobymulberry-博客園

1. 前言


這篇文章主要是對代理模式和委托模式進行了對比,個人認為Objective-C中的delegate大部分用法屬於委托模式。全文有些摳概念,對實際開發沒有任何影響。

前段時間看到的一篇博客iOS開發——從一道題看Delegate,和這篇博客iOS APP 架構漫談解決的問題類似。兩篇blog都寫得很不錯,都是為瞭解決兩個頁面之間的數據傳遞問題:

A頁面中有一個UILabel *labelA,B頁面中有一個UITextField *textFieldB。從A頁面跳轉到B頁面後,更改textFieldB中數據再返回到A頁面,labelA顯示的將是textFieldB中更改後的數據,嗯,就是這麼簡單的一個數據傳遞場景。

解決這個問題方法很多,比如使用一個DAO(data access object)去維護labelA和textFieldB所對應的數據。頁面的數據流向如下圖這樣:

image

但是這個場景不是很複雜,所以並不需要引入DAO這麼重的架構。

有時候我們會陷入技術的細節不可自拔,不妨靜下來想一想,這個問題本質在什麼?

這個問題的難點在於頁面B中textFieldB的數據變化後無法通知頁面A中的labelA。如果頁面B中有labelA的引用就好了,這樣就可以直接在頁面B的代碼中操作labelA。於是我在頁面B中添加了一個UILabel *labelARef,在A頁面push到B頁面時,將頁面A的labelA賦值給labelRef即可(親測可以進行數據傳遞)。

上述方法確實可行,不過大家肯定都覺得這樣設計也是太粗暴了。如果數據傳遞的業務比較多,那麼頁面B中就需要引用很多頁面A的屬性。當然我們可以直接引用頁面A作為頁面B的屬性,即UIViewController *vcA。如下圖所示:

image

這樣設計其實沒啥問題。不過我們這次主題是代理模式,那我們說的這個問題到底和代理模式有什麼聯繫呢?

2.使用代理模式實現數據傳遞


我們先看看GoF《設計模式:可復用面向軟體的基礎》中對代理模式的描述:為其他對象提供一種代理以控制對這個對象的訪問。咦,是不是和上面這個問題很像?為頁面B提供一種代理以控制頁面A的訪問,能控制頁面A,那就能控制頁面A中的labelA。可是上面那種直接引用對象的方法也可以提供對這個對象的訪問啊,為什麼一定要通過代理呢?我們來看下代理模式的UML圖:

image註意上圖中Proxy和RealSubject都實現了Subject這個介面,並且實現了相同的介面函數DoAction(),另外Proxy存有一份RealSubject的引用,即圖中的delegate。一般來說,Proxy在實現DoAction時,會調用RealSubject的DoAction,也就是利用所引用的delegate調用RealSubject的DoAction。按照我自己的理解,之所以會出現代理模式,是由於用戶需要對RealSubject的DoAction功能進行擴展,又無法對RealSubject中的DoAction直接進行修改(而且也違反了封閉-開放原則),於是使用了Proxy對RealSubject的DoAction進行了擴展,而擴展的內容都是DoAction,所以又將DoAction抽象出來,做成了介面。

回到上面那個案例,我們可以利用代理模式進行如下架構設計:

image

這裡介紹一個小技巧,即如何辨別誰是代理 —— 直接跟Client打交道的是代理,此處Client就是ViewControllerB的textFieldB控制項,所以直接打交道的就是ViewControllerB,也就是說ViewControllerB是代理。

代碼如下:

// DataTransDelegate

// DataTransDelegate
@protocol DataTransDelegate <NSObject>
- (void)didTextFieldChanged:(UITextField *)textField;
@end

// ViewControllerA

// ViewControllerA.m
#import "ViewControllerA.h"
#import "ViewControllerB.h"
#import "DataTransDelegate.h"

@interface ViewControllerA () <DataTransDelegate>
@property (strong, nonatomic) UILabel *labelA;
@property (strong, nonatomic) UIButton *buttonA;
@end

@implementation ViewControllerA

- (void)viewDidLoad {
    [super viewDidLoad];
    
    [self.view addSubview:self.labelA];
    [self.view addSubview:self.buttonA];
    
    [self.buttonA addTarget:self action:@selector(pushVC) forControlEvents:UIControlEventTouchUpInside];
}

- (void)pushVC
{
    ViewControllerB *vcB = [[ViewControllerB alloc] init];
    vcB.delegate = self;
    [self.navigationController pushViewController:vcB animated:NO];
}

- (void)didTextFieldChanged:(UITextField *)textField
{
    self.labelA.text = textField.text;
}

- (UILabel *)labelA
{
    if (_labelA == nil) {
        _labelA = [[UILabel alloc] initWithFrame:CGRectMake(100, 100, 200, 50)];
        _labelA.text = @"顯示vcB中的textField內容";
    }
    return _labelA;
}

- (UIButton *)buttonA
{
    if (_buttonA == nil) {
        _buttonA = [[UIButton alloc] initWithFrame:CGRectMake(100, 200, 100, 50)];
        _buttonA.backgroundColor = [UIColor blueColor];
        [_buttonA setTitle:@"進入vcB" forState:UIControlStateNormal];
    }
    return _buttonA;
}

@end

// ViewControllerB

// ViewControllerB.h
@protocol DataTransDelegate;

@interface ViewControllerB : UIViewController
@property (nonatomic, weak) id<DataTransDelegate> delegate;
@end

// ViewController.m
#import "ViewControllerB.h"
#import "DataTransDelegate.h"

@interface ViewControllerB () <UITextFieldDelegate, DataTransDelegate>
@property (strong, nonatomic) UITextField *textFieldB;
@end

@implementation ViewControllerB

- (void)viewDidLoad {
    [super viewDidLoad];
    
    [self.view addSubview:self.textFieldB];
    self.textFieldB.delegate = self;
}

- (void)textFieldDidEndEditing:(UITextField *)textField
{
    [self didTextFieldChanged:textField];
}

- (void)didTextFieldChanged:(UITextField *)textField
{
    [self.delegate didTextFieldChanged:textField];
}

- (UITextField *)textFieldB
{
    if (_textFieldB == nil) {
        _textFieldB = [[UITextField alloc] initWithFrame:CGRectMake(100, 100, 100, 50)];
        _textFieldB.text = @"輸入文字";
        _textFieldB.backgroundColor = [UIColor redColor];
    }
    return _textFieldB;
}

@end

效果如下:

1

3.關於代理模式誤解


其實到目前為止並沒有什麼異樣。關鍵是在大家對Objective-C的protocol使用上,一般是結合delegate使用的。大多數我們稱這種模式是代理模式,但是我覺得delegate更像是一種委托模式,而非真正意義上的代理,代理是proxy,而委托是delegate。另外,代理模式中代理和被代理者都需要繼承並實現同一個介面Subject,而我們使用delegate一般只需要讓其中一個類繼承並實現對應介面即可。

委托模式是軟體設計模式中的一項基本技巧。在委托模式中,有兩個對象參與處理同一個請求,接受請求的對象將請求委托給另一個對象來處理。其實上面的viewControllerB包含了viewControllerA的引用這種做法就是委托模式。

比如我們最為熟知的UITableView,就是一個典型的委托模式,它將tableView的中不變的部分封裝起來,將經常變化的部分委托給用戶自己處理,所以說UITableView就是一個delegator,而遵循UITableViewDelegate的那個類就是delegate,所以我們經常會在一個UIViewController中使用類似self.tableView.delegate = self這樣的表達;

大家可能會疑惑為什麼還需要使用UITableViewDelegate這種類似於Java中的interface?我個人理解是因為這樣方便統一介面,介面統一了,方便了用戶,因為只需要實現這幾個介面就可以了。

所以我們可以看到最開始提到的兩篇博客其實藉助了Objective-C中的protocol實現了的其實是委托模式。

如果非要說委托模式和代理模式什麼關係的話,我覺得代理模式應該算是一種特殊的委托模式。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 例如一個小項目:實現單詞本的添加單詞等功能 功能:不同的方式實現跨app之間數據的暴露與接收 暴露端app:實現單詞的添加(Word、Translate),增刪改查; 接收端app:模糊查詢,得到暴露端的數據。 暴露端主頁及佈局: 1、佈局: 主頁佈局:ListView、TextView(empty ...
  • 蜂鳴器的種類和工作原理 原理 蜂鳴器也成為PWM(脈衝寬度調製),基本原理是通過脈衝來控制蜂鳴器的打開和停止。它是利用微控制器的數字輸出來對模擬電路進行控制的一種非常有效的技術,廣泛應用於測量、通信、功率控制與變換等許多領域。所以,我們要對蜂鳴器進行操作,就是通過對TOUT[0]引腳的設置,即將其設 ...
  • 本文主要對 應用程式、庫、內核、驅動程式的關係 及 LED驅動的實現原理 進行簡要介紹 ...
  • 在《Android 生成xml文件》一文中使用流的形式寫入xml格式文件,但是存在一定的問題,那就是在簡訊內容中不能出現<>之類的括弧,本文使用xml序列化器來解決 xml序列化器對象 XmlSerializer xs = Xml.newSerializer();* 給序列化器設置輸出流 File ...
  • RISC(reduced instruction set computer)精簡指令集電腦 簡介 精簡指令集,是電腦中央處理器的一種設計模式,也被稱為RISC(Reduced Instruction Set Computer的縮寫)。[1] 這種設計思路對指令數目和定址方式都做了精簡,使其實現更 ...
  • POP縮放動畫 效果 源碼 https://github.com/YouXianMing/Animations 細節 1. 參數設置有技巧,可以參考如下所示(項目中的POPSpringParameterController): 2. 動畫效果是通過組合兩個動畫而來的,要註意設置代理: ...
  • Camera提供了一個叫做setParameters的方法幫助開發者設置相機的相關參數。 通過Camera的getParameters方法可以獲取到當前為相機設置的相關參數。 下麵簡單介紹下,視頻錄製會用到的幾個參數的用法。 一、設置PreviewSize,即視頻預覽大小,也即輸出到SurfaceV ...
  • 看了幾篇文章,因為文章很新手向,所以內容很繁瑣。故整理一下重點,寫了測試程式去瞭解幾個知識點,不討論基本概念。新博客wossoneri.com "傳送門" 非集合類對象的copy與mutableCopy 在非集合類對象中:對 對象進行 操作,是指針複製, 操作時內容複製;對 對象進行 和`mutab ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...