【原】谈谈对Objective

  • 时间:
  • 浏览:0
  • 来源:大发uu快3_uu快3规则_大发uu快3规则

本文转载请注明出处 —— 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所对应的数据。页面的数据流向如下图曾经:

假如有一天这人 场景有的是很繁复,可是 未必并能 引入DAO这么重的架构。

有以后 亲戚亲戚大家歌词 歌词 会陷入技术的细节不可自拔,不妨静下来想一想,这人 疑问本质在那先 ?

这人 疑问的难点在于页面B中textFieldB的数据变化后无法通知页面A中的labelA。不可能 页面B带有labelA的引用就好了,曾经就并能 直接在页面B的代码中操作labelA。于是我在页面B中再加了曾经UILabel *labelARef,在A页面push到B页面时,将页面A的labelA赋值给labelRef即可(亲测并能 进行数据传递)。

上述最好的办法 随便说说可行,不过亲戚亲戚大家歌词 歌词 肯定都随便说说曾经设计也是越深暴了。不可能 数据传递的业务比较多,这么页面B中就并能 引用可是 页面A的属性。当然亲戚亲戚大家歌词 歌词 并能 直接引用页面A作为页面B的属性,即UIViewController *vcA。如下图所示:

曾经设计随便说说没啥疑问。不过亲戚亲戚大家歌词 歌词 这次主题是代理模式,曾经们说的这人 疑问到底和代理模式有那先 联系呢?

2.使用代理模式实现数据传递


亲戚亲戚大家歌词 歌词 先看看GoF《设计模式:可复用面向软件的基础》中对代理模式的描述:为以后 对象提供五种代理以控制对这人 对象的访问。咦,是有的是和上边这人 疑问很像?为页面B提供五种代理以控制页面A的访问,能控制页面A,那就能控制页面A中的labelA。曾经上边那种直接引用对象的最好的办法 可是 需要 提供对这人 对象的访问啊,为那先 一定要通过代理呢?亲戚亲戚大家歌词 歌词 来看下代理模式的UML图:

注意上图中Proxy和RealSubject都实现了Subject这人 接口,假如有一天实现了相同的接口函数DoAction(),另外Proxy存有一份RealSubject的引用,即图中的delegate。一般来说,Proxy在实现DoAction时,会调用RealSubject的DoAction,也可是 利用所引用的delegate调用RealSubject的DoAction。按照我自己的理解,未必会冒出 代理模式,是不可能 用户并能 对RealSubject的DoAction功能进行扩展,又无法对RealSubject中的DoAction直接进行修改(假如有一天也违反了封闭-开放原则),于是使用了Proxy对RealSubject的DoAction进行了扩展,而扩展的内容有的是DoAction,可是 又将DoAction抽象出来,做成了接口。

回到上边那个案例,亲戚亲戚大家歌词 歌词 并能 利用代理模式进行如下下发:

这里介绍曾经小技巧,即咋样辨别谁是代理 —— 直接跟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(30, 30, 30, 30)];
        _labelA.text = @"显示vcB中的textField内容";
    }
    return _labelA;
}

- (UIButton *)buttonA
{
    if (_buttonA == nil) {
        _buttonA = [[UIButton alloc] initWithFrame:CGRectMake(30, 30, 30, 30)];
        _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(30, 30, 30, 30)];
        _textFieldB.text = @"输入文字";
        _textFieldB.backgroundColor = [UIColor redColor];
    }
    return _textFieldB;
}

@end

效果如下:

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实现了的随便说说是委托模式。

不可能 必须说委托模式和代理模式那先 关系一句话,我随便说说代理模式应该与非 五种特殊的委托模式。