代理是什么?什么时候用代理?
代理其实是一种设计模式。怎么使用代理?
2.1 代理使用的前三步(a、声明代理原型;b、声明代理变量;c、调用代理方法)
2.2 代理使用的后三步(a、<>声明实现代理;b、设置代理的值;c、实现代理方法) [前三步和后三步是对应的]
在新创建的项目 BLDemo3 中再创建一个BLDragonView 的类。先来看一下视频教学中运行应用后界面是如何的:
再看一下这个类的头文件 .h 其实还是很简单的:
#import <UIKit/UIKit.h>
@interface BLDragonView:UIView
@property(nonatomic, strong)UIImageView *dragonImageView;
@end
上面的彩图中两个前进后退button,蓝绿色背景,龙的图片在 实现文件中 .m 实现:
#import "BLDragonView.h"
@implementation BLDragonView
- (instancetype)initWithFrame:(CGRect)frame
{
self = [super iniWithFrame:frame];
if (self) {
self.backgroundColor = [UIColor cyanColor];
self.dragonImageView = [[UIImageView alloc] initWithFrame:CGRectMake(10, 10, 100, 100)];
self.dragonImageView.image = [UIImage imageNamed:@"dragon.png"];
self.dragonImageView.backgroundColor = [UIColor clearColor];
[self.view addSubview:self.dragonImageView]; // `为什么要搞一个 self.`
UIButton *backButton = [[UIButton alloc] initWithFrame:CGRectMake(10, 120, 50, 50)];
[backButton buttonWithType:UIButtonTypeCustom];
[backButton setImage:[UIImage imageName:@"back.png"] forState:UIControlStateNormal];
[backButton addTarget:self
action:@selector(backButtonClicked:)
forControlEvents:UIControlEventTouchUpInside];
[self.view addSubview:backButton];
UIButton *forwardButton = [UIButton buttonWithType:UIButtonTypeCustom];
forwardButton.frame = CGRectMake(10, 120, 50, 50);
[forwardButton setImage:[UIImage imageNamed:@"forward.png"] forState:UIControlStateNormal];
[forwardButton addTarget:self
action:@selector(forwardButtonClicked:)
forControlEvents:UIControlEventTouchUpInside];
[self.view addSubview:forwardButton];
}
return self;
}
里面两个button的方法是前进和后退,代码如下:
- (void)backButtonClicked:(id)sender
{
CGFloat x = _dragonImageView.frame.origin.x - 10;
if (x > 0) {
_dragonImageView.frame = CGRectMake(x, _dragonImageView.frame.origin.y, 100, 100);
}
}
- (void)forwardButtonClicked:(id)sender
{
CGFloat x = _dragonImageView.frame.origin.x + 10;
if (x < self.frame.size.width - 100) {
_dragonImageView.frame = CGRectMake(x, _dragonImageView.frame.origin.y, 100, 100);
}
}
程序一旦运行,就会运行到 ViewController.m 中:
#import "viewController.h"
#import "BLDragonView.h"
@interface ViewController ()
@end
@implementation Viewcontroller
- (void)viewDidLoad {
[super viewDidLoad];
BLDragonView *dragonView = [[BLDragonView alloc] initWithFrame:CGRectMake(0, 0, self.frame.size.width, 180)];
[self.view addSubview:dragonView];
UIButton *hideButton = [UIButton buttonWithType:UIButtonTypeCustom];
hideButton.frame = CGRectMake((self.view.frame.size.width - 50 ) / 2, dragonView.frame.origin.y + 180 + 100, 50, 50);
[hideButton setImage:[UIImage imageNamed:@"hide.png"] forState:UIControlStateNormal];
[self.view addSubview:hideButton];
[hideButton addTarget:self
acton:@selector(hideButtonClicked:)
forControlEvents:UIControlEventTouchUpInside];
// 下面指的是图片中最下方的紫色状态栏
_dragonStatusLabel = [[UILabel alloc] initWithFrame:CGRectMake(0, self.view.frame.size.height - 60, self.view.frame.size.width, 60)];
_dragonStatusLabel.backgroundColor = [UIColor purpleColor];
_dragonStatusLabel.numberOfLines = 0;
_dragonStatusLabel.font = [UIFont systemFontOfSize:15];
_dragonStatusLabel.textAlignment = NSTextAlignmentCenter;
_dragonStatusLabel.text = @"dragon's status";
_dragonStatusLabel.textColor = [UIColor whiteColor];
[self.view addSubView:_dragonStatusLabel];
}
- (void)hideButtonClicked:(id)sender
{
}
当我们希望点击前进按钮,或者后退按钮的时候,希望龙在前进或者后退的时候,dragonStatusLabel 的值可以变成 “前进” 或者 “后退”。其实,要实现这样的需求,就需要用到 代理 了。
因为我们的前进按钮的方法设置,和后退按钮的方法设置(包括按钮本身)都是在 另一个类 BLDragonView.m 中设置的。而需要改变的值的view 是在 ViewController.m 这个类中。因此在 BLDragonView 这个类中 button 有它自己的作用域,它是访问不到 _dragonStatueLabel 的。需要将在 BLDragonView.m 中 发生 的 backButtonClicked: 和 forwardButtonClicked: 告诉给 ViewControll 让它来实现改变值的方法。 这样需要让代理方法出现了,点击 BLDemo2 ,新建一个文件 Objective-C file。选择它的类型是 protocol , 命名为 BLDragonViewDelegate :
emsp;生成的文件不是一个类,仅仅是一个声明文件,相当于代理的第一步,声明代理原型,里面还需要声明代理方法:
#import <Foundation/Foundation.h>
@class BLDragonView;
@protocol BLDragonViewDelegate <NSObject>
- (void)backButtonClicked:(BLDragonView *)dragonView; // 为什么在代理方法中最好有参数 ,是因为也许将来执行这个代理方法的类,也许会创建多个 BLDragonView 类的对象,所以可以通过参数的tag来做更多更方便的操作。
- (void)forwardButtonClicked:(BLDragonView *)dragonView;
需要在 BLDragonView.m 中 导入 :#import "BLDragonViewDelegate.h"
为什么在上面的代码段中不使用 #import <BLDragonView.h>,而使用 @class BLDragonView .因为使用前者会造成相互引用。 接下来需要声明代理变量了,在 BLDragonView.h 中添加一个属性:
......
@property(nonatomic, strong)UIImageView *dragonImageView;
@property(nonatomic, weak)id<BLDragonViewDelegate> delegate;
// 一般代理的变量 ,我们都叫 delegate, 至于为什么前面是 id 是因为,id代表任意对象类型,我们不清楚以后这个代理最后是谁(哪个类)来实现,
// 有可能是 viewControll ,也有可能其它的类来实现。
// 当然也可以说所有的类无非都是 NSObject , 所以可以写成这样:
// @property(nonatomic, weak)NSObject *delegate; 这样也是可以的。
// 但是,一般还是按原方法写,显得更好,并且一般代理的属性特性都用 weak , 涉及到了循环引用的问题
@end
// 最终上面那行代码的意思是,使用这个变量的类成为有能力实现 BLDragonViewDelegate 中声明的两个方法的类。
然后需要做代理的第三步,当某件事情发生的时候,调用这个代理方法,在BLDragonView.m 中实现:
- (void)backButtonClicked:(id)sender
{
CGFloat x = _dragonImageView.frame.origin.x - 10;
if (x > 0) {
_dragonImageView.frame = CGRectMake(x, _dragonImageView.frame.origin.y, 100, 100);
}
[_delegate backButtonClicked:self]; // 这个时候代理这个变量是空的,需要在后面的过程中给它赋值(即到底是谁来收衣服)
}
- (void)forwardButtonClicked:(id)sender
{
CGFloat x = _dragonImageView.frame.origin.x + 10;
if (x < self.frame.size.width - 100) {
_dragonImageView.frame = CGRectMake(x, _dragonImageView.frame.origin.y, 100, 100);
}
[_delegate forwardButtonClicked:self];
}
这样代理使用的前三步就完成了。接下来就是收衣服的方法由谁去执行呢(老师的代理方法的生活化比喻)?这个就是代理的后三步。现在假如我们是在 viewController 里面真正的实现这个代理,那么第一步是在 viewController.h 的头文件中声明实现代理方法:
#import <UIKit/UIKit.h>
#import <BLDragonViewDelegate.h>
@interface ViewController:UIViewController<BLDragonViewDelegate> // 这样就说明了 ViewController 这个类具备了去实现 BLDragonViewDelegate.h 当中的两个方法(即代理方法)
{
UILabel *_dragonStatusLabel;
}
然后在 ViewController.m 实现文件中实现代理方法:
- (void)viewDidLoad {
......
_dragonStatusLabel.textColor = [UIColor whiteColor];
[self.view addSubview:_dragonStatusLabel];
}
- (void)backButtonClicked:(BLDragonView *)dragonView
{
CGFloat X = dragonImageView.frame.origin.x;
CGFloat Y = dragonImageView.frame.origin.y;
_dragonStatusLabel.text = [NSString stringWithFormat:@"后退,(x,y) = (%f, %f)", X, Y];
// _dragonStatusLabel 这个就相当于老师举的例子中的衣服,它的值将被改变。
// 更好的表示,甚至将坐标值的改变也表示出来
}
- (void)forwardButtonClicked:(BLDragonView *)dragonView
{
CGFloat X = dragonImageView.frame.origin.x;
CGFloat Y = dragonImageView.frame.origin.y;
_dragonStatusLabel.text = [NSString stringWithFormat:@"前进,(x,y) = (%f, %f)", X, Y];
}
- (void)hideButtonClicked:(id)sender
{
}
但是这样还是不能实现我们想要的效果的,因为在代理的后三步中,我们还没有实现第二步: 设置代理的值(即到底是谁来执行,或者说是谁来收衣服呢,是妈妈还是女儿?)。那么指定的步骤应该在哪里实现呢?
如果是要让 ViewController 这个类来实现,就在它的 .m 文件中的 - (void)viewDidLoad {} 方法中输入:
- (void)viewDidLoad {
[super viewDidLoad];
BLDragonView *dragonView = [[BLDragonView alloc] initWithFrame:CGRectMake(0, 0, self.frame.size.width, 180)];
dragonView.delegate = self; // 这样就是设置了代理的值
[self.view addSubview:dragonView];
......
_dragonStatusLabel.textColor = [UIColor whiteColor];
[self.view addSubview:_dragonStatusLabel];
}