NSTimer的循环引用问题解决

作者: 图长伴 | 来源:发表于2017-06-22 17:40 被阅读0次

    场景:一个VC想引用一个带着NStimer的View.但是一般的情况下我们要在VC的dealloc中还要销毁View中的timer.如果不这样做view就不会释放,引起内存泄漏.甚至crash.如果多个VC引用这个View,那样每个VC的dealloc都要管理这个View的timer岂不是很麻烦.这时候我们肯定是想让View自己处理自己的timer.

    1.首先我们说一下NStimer循环引用的情况.我们直接看代码.

    self.timer = [NSTimer timerWithTimeInterval:self.animationDuration target:self selector:@selector(switchPage) userInfo:nil repeats:YES];

    [[NSRunLoop currentRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];

    这样就会造成一个这样的循环引用:

    图1

    想要打破这种循环只有两种办法.

    1.控制器不再引用定时器

    2.定时器不再保留当前控制器

    第一种是不可能的了,那就只能是第二种了.接下来我们看看怎么打破这种循环.

    首先我们要认识一下NSProxy这个类.

    NSProxy是一个抽象类,它为一些表现的像是其它对象替身或者并不存在的对象定义一套API。一般的,发送给代理的消息被转发给一个真实的对象或者代理本身load(或者将本身转换成)一个真实的对象。NSProxy的基类可以被用来透明的转发消息或者耗费巨大的对象的lazy初始化。NSProxy实现了包括NSObject协议在内基类所需的基础方法,但是作为一个虚拟的基类并没有提供初始化的方法。它接收到任何自己没有定义的方法他都会产生一个异常,所以一个实际的子类必须提供一个初始化方法或者创建方法,并且重载forwardInvocation:方法和methodSignatureForSelector:方法来处理自己没有实现的消息。一个子类的forwardInvocation:实现应该采取所有措施来处理invocation,比如转发网络消息,或者加载一个真实的对象,并把invocation转发给他。methodSignatureForSelector:需要为给定消息提供参数类型信息,子类的实现应该有能力决定他应该转发消息的参数类型,并构造相对应的NSMethodSignature对象。详细信息可以查看NSDistantObject,NSInvocation, and NSMethodSignature的类型说明。

    说白了就是做一个消息的转发.

    我们要做的就是实例一个NSProxy的subclass.让NSTimer定时中的方法由NSProxy的subclass转发给View执行.但是NStimer持有的却不是View.这样就不会循环引用的.

    首先我们构建NSProxy的subclass.命名为VMRProxy.

    VMRProxy.h中的代码

    #import@interface VMRProxy : NSProxy

    @property (nonatomic, weak) id target;

    @end

    VMRProxy.m中的代码

    #import "VMRProxy.h"

    @implementation VMRProxy

    - (NSMethodSignature *)methodSignatureForSelector:(SEL)sel {

    return [self.target methodSignatureForSelector:sel];

    }

    - (void)forwardInvocation:(NSInvocation *)invocation {

    [invocation invokeWithTarget:self.target];

    }

    @end

    NSProxy的subclass必须重写接两个方法.文档上是这样说的:

    A concrete subclass must therefore provide an initialization or creation method and override theforwardInvocation:andmethodSignatureForSelector:methods to handle messages that it doesn’t implement itself.

    接下来我们看看View中是怎样实现的:

    在View的.m文件中构建属性:

    @property (nonatomic, strong) NSTimer *timer;

    @property (nonatomic, strong) VMRProxy *proxy;

    // 实例proxy 并指定执行方法的对象

    - (instancetype)initWithFrame:(CGRect)frame {

    self = [super initWithFrame:frame];

    if (self) {

    self.proxy = [VMRProxy alloc];

    self.proxy.target = self;

    }

    return self;

    }

    // 使用定时器

    self.timer = [NSTimer timerWithTimeInterval:self.animationDuration target:self.proxy selector:@selector(switchPage) userInfo:nil repeats:YES];

    [[NSRunLoop currentRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];

    这块代码可以放在initWithFrame:方法中.也可以写在自定义的方法中,在initWithFrame:
    中引用自定义的方法,毕竟这样看着比较清晰吧.(我的代码习惯会这么做)

    这样View的dealloc方法就会调用了.然后在dealloc中做这样的操作.

    - (void)dealloc {

    [self.timer invalidate];

    self.timer = nil;

    }

    一个能自己销毁定时器的view就做成了.

    注意事项:

    1.VMRProxy 一定要定义成全局的属性.要是局部的话,出了}就会销毁了.

    2.NSProxy 还有好多用处,建议大家在深度学一学.

    大概就这些了,有哪些写的不好的欢迎大家指正.多谢!

    相关文章

      网友评论

        本文标题:NSTimer的循环引用问题解决

        本文链接:https://www.haomeiwen.com/subject/hdyfcxtx.html