- 说明:通过一个废弃的alertView,详解运行时关联的一个巧妙的实例
基础知识:
- 用runtime需要导入头文件
#import "objc/runtime.h"
- 方法调用的本质,就是让对象发送消息
- objc_msgSend,只有对象才能发送消息,因此以objc开头
- 给一个分类添加属性,其实本质就是给这个分类添加关联,并不是直接把这个值的内存空间添加到类存空间
- 运行时参数配置PROJECT ->Build Settings ->搜索msg ->Enable Strict Checking of objc_msgSend Calls ->更改为No
应用场景
- 如果一个页面调用了多次alertView弹窗,根据alertViewDelegate代理回调,处理不同的业务,那么如何辨别是哪个alertView
- 一个简单的方法就是给不同的alertView设置不同的tag,在alertViewDelegate代理回调发放中,根据不同的tag,if判断一大推;
- 另外一个解决方法,利用运行时,给alertView扩充一个分类,声明一个传入block参数的方法,设置alertView分类作为alertViewDelegate的代理,并且实现代理方法,利用运行时,拿到传过来的block,并执行,那么根据分类优先原则,会优先调用分类的方法,在alertView退出界面后,断开管理;
直接上代码
#import <UIKit/UIKit.h>
#import "objc/runtime.h"
typedef void(^blockCallBack)(NSInteger index);
@interface UIAlertView (BlockSupport)<UIAlertViewDelegate>
-(void)setCallBackBlock:(blockCallBack)block;
@end
1.设置关联对象
-(void)setCallBackBlock:(blockCallBack)block
{
// static char kAssociatedObjectKey;
// 将某个值跟某个对象关联起来,将某个值存储到某个对象中
// key:通常推荐key使用static char类型——使用指针或许更好,key值是一个唯一的常量,并只在getters和setters方法内部使用objc_getAssociatedObject(self, &kAssociatedObjectKey);
// 一个更简单的方案是:直接使用选择器(selector)因为SEL生成的时候就是一个唯一的常量,你可以使用 _cmd 作为objc_setAssociatedObject()的key
// value:需要保存的值,这里传block,把block先保存起来,合适的地方调用
// policy:这里的policy跟属性声明中的retain、assign、copy是一样的,不再赘述
// objc_setAssociatedObject(<#id object#>, <#const void *key#>, <#id value#>, <#objc_AssociationPolicy policy#>)
objc_setAssociatedObject(self, @selector(setCallBackBlock:), block, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
self.delegate=self;
}
2. 获取相关联的对象,执行block
-(void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex
{
blockCallBack block=objc_getAssociatedObject(self, @selector(setCallBackBlock:));
!block ? : block(buttonIndex);
}
3.断开关联
-(void)alertView:(UIAlertView *)alertView didDismissWithButtonIndex:(NSInteger)buttonIndex
{
// 移除关联对象 不恰当的做法
// 根据苹果文档描述,你不大可能有需求要自己去调用
// 因为它也会移除掉包括其他地方加入的全部的关联对象
// objc_removeAssociatedObjects(self);
// 一般你只需要通过调用objc_setAssociatedObject并传入nil值来清除关联值
objc_setAssociatedObject(self, @selector(setCallBackBlock:), nil, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
4.用法示例
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"测试提示" message:@"这是船长的demo" delegate:self cancelButtonTitle:@"取消" otherButtonTitles:@"确定", nil];
[alert setCallBackBlock:^(NSInteger index) {
if (index != alert.cancelButtonIndex) {
NSLog(@"点击了确定按钮");
}
}];
[alert show];
}
- 举一反三,大家也可以联想到
UIActionSheet
等其他控件,给他们也写个分类,方便我们使用