高效开发iOS系列(runtime篇) -- 巧妙运用对象关联(Associated Objects)来设计alertview

原创 2015年12月22日 22:15:15

热身 (简单介绍)


这个鬼(objc_AssociationPolicy : 对象关联)可以给某对象关联其他对象, 这些对象通过"键"来区分. 存储对象值的时候, 可以指明"存储策略", 用以维护相应的"内存管理语义".存储策略由名为objc_AssociationPolicy的枚举所定义. 同时拥有与之等效@property的属性. … (此处省略十万字的理论)

本篇重点是讲述如何巧妙利用这个东西(关联), 来让我们高效的编写objective-c. 也可以理解为实战篇.

高效开发iOS系列里, 大部分都是偏向于实战的技巧方法与总结来提升iOS开发效率, 几乎都不讲理论,所以需要具备一点基础. 而本篇讲的则是objc_AssociationPolicy(关联)alertview之间的实战技巧, 如果还不会Associated Objects 或者想要更深入更全面的认识Associated Objects理论的小伙伴们, 可以查看这篇理论篇文章入门 : http://blog.csdn.net/biggercoffee/article/details/50135477

来一起干一票


繁琐的alertview的事件处理, 早就看它不爽了

// button 点击执行的方法
- (IBAction)alertClick:(id)sender {
    UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"title" message:@"message" delegate:self cancelButtonTitle:@"cancel" otherButtonTitles:@"ok", nil];
    [alertView show];
}
// alertview 点击执行的delegate方法,前提 需要self实现UIAlertViewDelegate协议
- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex {
    NSLog(@"执行逻辑操作");
}


如上代码, 这样写确实没错, 但是如果一个页面有多个alertview, 而且点击的时候都要执行不同的逻辑尼? 则要在alertview的delegate方法中写很多的if else 语句, 可读性,扩展性等等都非常差, 如下代码示例:

// button1 点击执行的方法
- (IBAction)alertClick:(id)sender {
    UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"title" message:@"message" delegate:self cancelButtonTitle:@"cancel" otherButtonTitles:@"ok", nil];
    alertView.tag = 1000;
    [alertView show];
}

// button2 点击执行的方法
- (IBAction)alertClick:(id)sender {
    UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"title" message:@"message" delegate:self cancelButtonTitle:@"cancel" otherButtonTitles:@"ok", nil];
    alertView.tag = 2000;
    [alertView show];
}

// button3 点击执行的方法 ...等等很多alertview
- (IBAction)alertClick:(id)sender {
    UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"title" message:@"message" delegate:self cancelButtonTitle:@"cancel" otherButtonTitles:@"ok", nil];
    alertView.tag = 3000;
    [alertView show];
}
// alertview 点击执行的delegate方法,前提 需要self实现UIAlertViewDelegate协议
- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex {
    if (alertView.tag == 1000) {
        NSLog(@"alertView1 执行逻辑操作");
    }
    else if (alertView.tag == 2000) {
        NSLog(@"alertView2 执行逻辑操作");
    }
    else if (alertView.tag == 3000) {
        NSLog(@"alertView3 执行逻辑操作");
    }
    // ...等等很多alertview
}


上面这一大串代码不用解释, 一眼看着都不爽.在delegate的方法中根据tag或者成员变量的alert等很多方式来标注alertview的唯一性, 然后在根据不同的alertview和点击不同的button来执行逻辑,一堆if else太繁琐了.

block来代替delegate,现在看如下代码示例 :


// button1 点击执行的方法
- (IBAction)alertClick:(id)sender {
    UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"title" message:@"message" delegate:nil cancelButtonTitle:@"cancel" otherButtonTitles:@"ok", nil];
    [alertView show];
    [alertView zxp_clickedButtonAtIndexWithBlock:^(UIAlertView *alertView, NSInteger buttonIndex) {
        NSLog(@"---%zi",buttonIndex);
    }];
}

// button2 点击执行的方法
- (IBAction)alertClick:(id)sender {
    UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"title" message:@"message" delegate:nil cancelButtonTitle:@"cancel" otherButtonTitles:@"ok", nil];
    [alertView show];
    [alertView zxp_clickedButtonAtIndexWithBlock:^(UIAlertView *alertView, NSInteger buttonIndex) {
        NSLog(@"---%zi",buttonIndex);
    }];
}

// button3 点击执行的方法 
- (IBAction)alertClick:(id)sender {
    UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"title" message:@"message" delegate:nil cancelButtonTitle:@"cancel" otherButtonTitles:@"ok", nil];
    [alertView show];
    [alertView zxp_clickedButtonAtIndexWithBlock:^(UIAlertView *alertView, NSInteger buttonIndex) {
        NSLog(@"---%zi",buttonIndex);
    }];
}

开始解剖 -> 三步走



第一步: 创建一个alertview 的category, 并提供与delegate方法一样操作方式的apis

第二步: 在category内部把delegate交给其他class来管理并把block也传递给delegate管理的class

第三步: 管理delegate的class调用block即可

示例代码:


.h文件


@interface UIAlertView (zxp_block)

- (void)zxp_clickedButtonAtIndexWithBlock:(void(^)(UIAlertView *alertView,NSInteger buttonIndex))block;


@end

.m文件


#import <objc/runtime.h>

static const char *ZXPDelegateKey = "!ZXPDelegateKey!";

#pragma mark - class
@interface ZXPAlertViewDelegate : NSObject <UIAlertViewDelegate>
{
    //block变量
    void(^_block)();
}

//初始化的时候传入一个block
- (instancetype)initWithBlock:(void(^)())block;

@end

@implementation ZXPAlertViewDelegate

- (instancetype)initWithBlock:(void(^)())block
{
    self = [super init];
    if (self) {
        //设置关联对象, 值为self
        objc_setAssociatedObject(self, ZXPDelegateKey, self, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
        _block = block;
    }
    return self;
}

#pragma mark - delegate methods

- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex {

    if (_block) {
        //执行block
        _block(alertView,buttonIndex);
    }

}

@end

#pragma mark - category



@implementation UIAlertView (zxp_block)

- (void)zxp_clickedButtonAtIndexWithBlock:(void(^)(UIAlertView *alertView,NSInteger buttonIndex))block {
    //获取delegate对象
    id delegate = objc_getAssociatedObject([[ZXPAlertViewDelegate alloc] initWithBlock:block], ZXPDelegateKey);
    //设置delegate为其他对象
    self.delegate = delegate;

}

@end
版权声明:本文为博主原创文章,转载请注明出处。

相关文章推荐

LocationManager定位国内偏移,WGS-84、GCJ-02、BD-09之间转换

// CoordinateTransformation.h// Created by MLS on 15/12/22. // Copyright © 2015年 MLS. All rights ...

圆形进度条,可设置内外环,多彩

今天做一个健身的项目,客户需要一个圆形的进度条来查看个人的运动记录等,做了一些功课之后贴出代码大家一起分享下 新建一个继承UIView类 .h文件 @interface MyProgressVie...

objc/runtime.h之Associated Objects ==关联

Objective-C #import Objective-C开发者应该小心谨慎地遵循这个危险咒语的各种准则。一个很好的原因的就是:混乱的运行时代码会改变运行在其架构之上的所有...

深入学习Objective-C(二)理解 objc 关联对象 (Associated Objects)

我们都知道,我们在普通的 objc 类中,一般我们都会把成员变量声明在@interface中,如果你想把成员变量暴露在头文件中,你可以把它声明在实现文件中,甚至你也可以放在类扩展的区域中,但是,你却不...
  • lkeplei
  • lkeplei
  • 2015年03月27日 19:12
  • 468

深入学习Objective-C(二)理解 objc 关联对象 (Associated Objects)

今天看了下 objc 2.0 引入的强大特性:关联对象,下面把我的理解与大家分享一下。 我们都知道,我们在普通的 objc 类中,一般我们都会把成员变量声明在@interface中,如果你想把成...
  • Dev_Ho
  • Dev_Ho
  • 2014年11月03日 23:00
  • 6319

runtime之Associated Objects

#import Objective-C开发者应该小心谨慎地遵循这个危险咒语的各种准则。一个很好的原因的就是:混乱的运行时代码会改变运行在其架构之上的所有代码。 从利的角度来讲,  中的函数...

iOS开发------runtime关联对象(动态添加属性)

如果想用一个系统的类,但是该类又不满足需求,首先肯定是继承自系统的类,重新构建一个类,但这种方法,如果我们需求的类之比原来的类多一个属性呢,继承似乎又显得比较麻烦,这个时候也许用runtime的关联对...

高效开发iOS系列(runtime篇) -- debug模式下使用po命令打印model的属性和值(runtime + debugDescription)

在开发过程中, 往往会有很多的model来装载属性. 而在开发期间经常会进行调试查看model里的属性值是否正确. 那么问题来了, 在`objective-c`里使用`NSLog("%@",model...

XZ_iOS之Runtime使用运行时实现字典转模型和关联对象

使用运行时实现字典转模型和关联对象objc_get/setAssociatedObject() 创建一个类Person,重写它的description方法,打印出它的属性值。 XZPerson...

iOS 利用runtime关联对象

有次在大牛群看到一个问题:“如何给一个字典添加一个属性(不能继承)”,立马蒙逼了,不能用继承,难道用分类?但是分类貌似只能添加方法不能添加属性啊,百思不得其解,直到后来接触到了runtime才恍然大悟...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:高效开发iOS系列(runtime篇) -- 巧妙运用对象关联(Associated Objects)来设计alertview
举报原因:
原因补充:

(最多只允许输入30个字)