不少第三方框架中,有这样的一种用法:有一个.h的头文件,它里面只有协议的声明,并且没有对应的.m实现文件。一般这种文件就用来专职声明协议的做用了。git
这里举一个你们都能看到和下载的例子,优秀的第三方图文混排框架:TYAttributedLabel。github
这个框架里面有这样一段代码:bash
// 添加响应点击rect
- (void)addRunRectDictionary:(NSDictionary *)runRectDictionary
{
if (runRectDictionary.count < _runRectDictionary.count) {
NSMutableArray *drawStorageArray = [[_runRectDictionary allValues]mutableCopy];
// 剔除已经画出来的
[drawStorageArray removeObjectsInArray:[runRectDictionary allValues]];
// 遍历不会画出来的
for (id<TYTextStorageProtocol>drawStorage in drawStorageArray) {
if ([drawStorage conformsToProtocol:@protocol(TYViewStorageProtocol)]) {
[(id<TYViewStorageProtocol>)drawStorage didNotDrawRun];
}
}
}
_runRectDictionary = runRectDictionary;
}
复制代码
能够先关注这个:框架
if ([drawStorage conformsToProtocol:@protocol(TYViewStorageProtocol)])
复制代码
这行代码主要作一个是否服从协议的判断。这个协议是声明在TYTextStorageProtocol.h文件里面的。你们查看这个文件可知,它里面主要声明了几个协议,并且没有对应的.m文件。ui
打开TYTextStorageProtocol.h查看协议的代码:atom
@protocol TYViewStorageProtocol <NSObject>
/**
* 设置所属的view
*/
- (void)setOwnerView:(UIView *)ownerView;
/**
* 不会把你绘画出来
*/
- (void)didNotDrawRun;
@end
复制代码
[(id<TYViewStorageProtocol>)drawStorage didNotDrawRun];
复制代码
这句话里面的didNotDrawRun
方法是前面那个协议声明的方法,因此,只有当前面作了判断是否服从该协议以后,才可将暂时仍是个id类型的drawStorage对象进行转换,转成一个服从该协议的对象,进而执行协议方法。spa
对象的转换:.net
(id<TYViewStorageProtocol>)drawStorage
复制代码
至于一个对象是否“服从”的标准:就是该对象的类或其父类的@interface后面跟上一个协议,并实现协议的方法。code
接下来再这里的drawStorage对象,是怎样选择“服从” 协议TYViewStorageProtocol
的:orm
#import "TYDrawStorage.h"
@interface TYViewStorage : TYDrawStorage<TYViewStorageProtocol>
@property (nonatomic, strong) UIView *view; // 添加view
@end
复制代码
//TYViewStorageProtocol
- (void)didNotDrawRun
{
[_view removeFromSuperview];
}
复制代码
上面的协议声明有这样一行:
@protocol TYViewStorageProtocol <NSObject>
复制代码
这是协议的继承用法,该自定义协议继承自 根协议 ,协议的继承具备多继承的特色。
要注意的是,这里的 根协议 与常见的 基类 NSObject是两种概念:一个是类,一个是协议。NSObject类服从NSObject协议。具体的关系可参考笔者的另外一篇 iOS·NSObject的两种含义:类与协议