美文网首页iOS 宝典iOS Developer
iOS delegate继承(协议、代理的继承)

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

作者: 我在敲BUG | 来源:发表于2017-06-14 15:03 被阅读992次

    之前一直很好奇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;
    。。。
    

    相关文章

      网友评论

      本文标题:iOS delegate继承(协议、代理的继承)

      本文链接:https://www.haomeiwen.com/subject/osnsqxtx.html