场景:一个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 还有好多用处,建议大家在深度学一学.
大概就这些了,有哪些写的不好的欢迎大家指正.多谢!
网友评论