很多第三方框架中,有这样的一种用法:有一个.h的头文件,它里面只有协议的声明,而且没有对应的.m实现文件。通常这种文件就用来专职声明协议的作用了。
这里举一个大家都能看到和下载的例子,优秀的第三方图文混排框架:TYAttributedLabel。
这个框架里面有这样一段代码:
// 添加响应点击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文件。
打开TYTextStorageProtocol.h查看协议的代码:
@protocol TYViewStorageProtocol <NSObject>
/**
* 设置所属的view
*/
- (void)setOwnerView:(UIView *)ownerView;
/**
* 不会把你绘画出来
*/
- (void)didNotDrawRun;
@end
[(id<TYViewStorageProtocol>)drawStorage didNotDrawRun];
这句话里面的didNotDrawRun
方法是前面那个协议声明的方法,所以,只有当前面做了判断是否服从该协议之后,才可将暂时还是个id类型的drawStorage对象进行转换,转成一个服从该协议的对象,进而执行协议方法。
对象的转换:
(id<TYViewStorageProtocol>)drawStorage
至于一个对象是否“服从”的标准:就是该对象的类或其父类的@interface后面跟上一个协议,并实现协议的方法。
接下来再这里的drawStorage对象,是怎样选择“服从” 协议TYViewStorageProtocol
的:
#import "TYDrawStorage.h"
@interface TYViewStorage : TYDrawStorage<TYViewStorageProtocol>
@property (nonatomic, strong) UIView *view; // 添加view
@end
//TYViewStorageProtocol
- (void)didNotDrawRun
{
[_view removeFromSuperview];
}
区分class_conformsToProtocol与conformsToProtocol https://blog.csdn.net/likendsl/article/details/7567719