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