高效开发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
版权声明:本文为博主原创文章,转载请注明出处。

相关文章推荐

Java判断系统是否装了获正在运行360(某种软件)

通过判断进程判断系统正在运行某个软件package aa; import java.io.BufferedReader; import java.io.IOException; im...

HBuilder常用快捷键总结

HBuilder常用快捷键大概共9类(【4 13 3】文件、编辑、插入;【4 9 8】选择、跳转、查找;【1 1 6】运行、工具、视图)1.文件(4) 新建 ...

iOS进阶之runtime(2)--KVO简单模拟实现

前言: 上篇简单介绍了下runtime的几个API和KVO的底层原理,现在开始进入正题,来利用这几个API简单的模拟下KVO的原理.大概的步骤就是. 1.在注册KVO的时候注册一个通知,并...

某日与三三君的聊天记录

我觉得本地主要就是跟内蒙农大合作,找农牧厅的指导在高校 文化单位,政府单位搞点马文化草原文化的征文,还有马摄影展, 爱心绝骡  21:20:24活动就得找 俱乐部的,马术学校的, 雷克萨斯 21:20...

『干货』分享你最喜欢的技巧和提示(Xcode,objective-c,swift,c...等等)

亲爱的读者们,你们好 !年底将近,分享从过去一年你最喜欢的技巧和建议作为礼物送给新手们。提交你的最喜欢的迅速或objc琐事,有用的提示,意外的发现,有用的解决方法,无用的迷恋,或任何其他你觉得今年很酷...

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

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

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

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

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

我们都知道,我们在普通的 objc 类中,一般我们都会把成员变量声明在@interface中,如果你想把成员变量暴露在头文件中,你可以把它声明在实现文件中,甚至你也可以放在类扩展的区域中,但是,你却不...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

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