iOS-运行时(关联详解实例)

  • 说明:通过一个废弃的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 
#import "objc/runtime.h"
typedef void(^blockCallBack)(NSInteger index);
@interface UIAlertView (BlockSupport)
-(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 *)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等其他控件,给他们也写个分类,方便我们使用

你可能感兴趣的:(iOS-运行时(关联详解实例))