iOS 宝典iOS Developer

iOS delegate继承(协议、代理的继承)

2017-06-14  本文已影响992人  我在敲BUG

之前一直很好奇UITableView有一个delegate属性,遵守 <UITableViewDelegate>协议的 ,而它的父类UIScrollView也有一个同名的delegate,但是遵守的是<UIScrollViewDelegate>协议。这是怎么实现的呢?

先上代码

.h文件

#import <UIKit/UIKit.h>
@class CustomTextView;

// 创建自己的协议,同时遵守于父类协议(也就相当于继承子父类协议,实际上协议本身是不能被继承的,遵守就相当于抄了一份过来吧)
@protocol CustomTextViewDelegate <UITextViewDelegate>
// 这里添加自己在原协议方法基础上,新增的一些协议方法
// 示例:
- (void)customTextView:(CustomTextView *)customTextView someNewAction:(BOOL)action;
@end

@interface CustomTextView : UITextView
// 因为delegate此时要遵守我们新创建的协议,而不是原本的协议,所以要重写父类中的属性(这里会有警告,稍后会在.m文件中消除)
@property (nonatomic, weak) id<CustomTextViewDelegate> delegate;
@end

.m文件

#import "CustomTextView.h"

@implementation CustomTextView
@dynamic delegate;// .h中警告说delegate在父类已经声明过了,子类再声明也不会重新生成新的方法了。我们就在这里使用@dynamic告诉系统delegate的setter与getter方法由用户自己实现,不由系统自动生成

#pragma Set & Get
- (void)setDelegate:(id<CustomTextViewDelegate>)delegate
{
  super.delegate = delegate;
}

- (id<CustomTextViewDelegate>)delegate
{
  id curDelegate = super.delegate;
  return curDelegate;
}

// 找个地方触发下新加的协议方法,看是否正常工作
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
  {
    if ([self.delegate respondsToSelector:@selector(customTextView:someNewAction:)]) {
      [self.delegate customTextView:self someNewAction:YES];
  }
}

上面就是最简单的:继承父类协议,然后在其基础上增加新的协议方法

然而我们更常见的需求是,需要对父类中的协议方法做修改,比如在父类的某协议调用前插入一些操作,然后再调用代理(或者在调用父类某协议后,再插入一些操作)

思路是:将父类的delegate指向自己,自己实现父类中的协议方法,并在其中做修改,然后在适当的时间调用自己的delegate,将事件调用传递出去

通俗些说:有个作者告诉编辑说:“我的文章写好了”(通过作者的delegate告诉编辑),然后编辑说:“我要校审下这个文章,改一改”(编辑实现了作者的delegate中的方法,在其内做修改),最后编辑告诉一个读者:“文章改好了,你可以去读了”(最后通过编辑的delegate将事件传递出去)

需要修改父类协议的做法

.h文件的代码上面一样

#import <UIKit/UIKit.h>
@class CustomTextView;

// 创建自己的协议,同时遵守于父类协议(也就相当于继承子父类协议,实际上协议本身是不能被继承的,遵守就相当于抄了一份过来吧)
@protocol CustomTextViewDelegate <UITextViewDelegate>
// 这里添加自己在原协议方法基础上,新增的一些协议方法
// 示例:
- (void)customTextView:(CustomTextView *)customTextView someNewAction:(BOOL)action;
@end

@interface CustomTextView : UITextView
// 因为delegate此时要遵守我们新创建的协议,而不是原本的协议,所以要重写父类中的属性(这里会有警告,稍后会在.m文件中消除)
@property (nonatomic, weak) id<CustomTextViewDelegate> delegate;
@end

.m文件

#import "CustomTextView.h"

@interface CustomTextView () <UITextViewDelegate> // 这里要遵守父类原本的协议,因为继承类时,协议是不会被继承的,所以需要重新声明
@property (nonatomic, weak) id<CustomTextViewDelegate> myDelegate;// 因为父类的delegate对象是self,self的delegate是外部类,所以需要新增一个myDelegate来保存外部类
@end

@implementation CustomTextView
@dynamic delegate;// .h中警告说delegate在父类已经声明过了,子类再声明也不会重新生成新的方法了。我们就在这里使用@dynamic告诉系统delegate的setter与getter方法由用户自己实现,不由系统自动生成

#pragma mark - Set & Get
- (void)setDelegate:(id<CustomTextViewDelegate>)delegate
{
    super.delegate = self;// 这句也可以放在初始化方法中
    _myDelegate = delegate;
}

- (id<CustomTextViewDelegate>)delegate
{
    return _myDelegate;
}

#pragma mark - UITextViewDelegate
- (BOOL)textViewShouldBeginEditing:(UITextView *)textView
{
    BOOL result = NO;

    /*
    调用前可以做一些操作
    。。。
    */

    // 将协议事件传递出去
    if ([_myDelegate respondsToSelector:@selector(textViewShouldBeginEditing:)]) {
        result = [_myDelegate textViewShouldBeginEditing:self];
    }

    /*
     调用后也可以做一些操作
     。。。
    */

    // 可以在达到某个条件的情况下,调用新增的协议方法
    if ([_myDelegate respondsToSelector:@selector(customTextView:someNewAction:)]) {
        [_myDelegate customTextView:self someNewAction:YES];
    }

    return result;
 }

/*
 建议将原协议的所有方法都实现,不需要插入操作的就直接用_myDelegate传递出去
 */
- (BOOL)textViewShouldEndEditing:(UITextView *)textView
{
  BOOL result = NO;
  if ([_myDelegate respondsToSelector:@selector(textViewShouldEndEditing:)]) {
      result = [_myDelegate textViewShouldEndEditing:self];
  }
  return result;
}

// 剩余方法类似
- (void)textViewDidBeginEditing:(UITextView *)textView;
- (void)textViewDidEndEditing:(UITextView *)textView;
。。。
上一篇下一篇

猜你喜欢

热点阅读