打破NSTimer循环引用的方案

2019-04-23  本文已影响0人  AndreaArlex

NSTimer相信大家经常都会用到,但是,如果不注意的话,很容易就会造成内存泄漏。
我们先来暂时一个例子:

@interface HomeControllerViewController ()

@property(nonatomic, strong)NSTimer *timer;

@end

@implementation HomeControllerViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    // Do any additional setup after loading the view.
    
    self.view.backgroundColor = [UIColor redColor];
    
    self.timer = [NSTimer timerWithTimeInterval:1 target:self selector:@selector(timerSchedue:) userInfo:nil repeats:YES];
    [[NSRunLoop mainRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];
}

- (void)timerSchedue:(NSTimer *)timer {
    
    static int i = 0;
    NSLog(@"%d",i);
    i++;
}

- (void)dealloc {
    
    NSLog(@"已经销毁HomeControllerViewController");
}

@end

如果你从A controller推进一个Homecontroller,然后再点返回,那么造成的现象就是,一直打印1,2,3......不会停止,直到你杀掉进程。并且,NSLog(@"已经销毁HomeControllerViewController");这句不会打印。

原因分析
为什么会这样呢?
可能有人会猜:既然不走delloc方法,那肯定就是有循环引用在,导致不能释放HomeControllerViewController,那么,我把timer的target传weakSelf,会不会解决这个问题呢?
经过试验之后,答案是,并不能,还是不走delloc方法。

那么,我们要怎么解决这个问题呢?
我们先来看看文档:


image.png

文档上有提到,The receiver retains a Timer。很明显,接收者会强引用timer,那么你穿weakSelf就没什么用了。

image.png

如果我们采取一个proxy类,把他的引用断开呢?


image.png

这样是否可以解决这个问题呢?按原理来说是应该可以的。我们来尝试下:
我们先创建一个weakProxy类:

#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface WeakProxy : NSProxy

@property (weak,nonatomic,readonly)id target;

+ (instancetype)proxyWithTarget:(id)target;
- (instancetype)initWithTarget:(id)target;

@end

NS_ASSUME_NONNULL_END


#import "WeakProxy.h"

@implementation WeakProxy

- (instancetype)initWithTarget:(id)target {
    _target = target;
    return self;
}

+ (instancetype)proxyWithTarget:(id)target {
    return [[WeakProxy alloc] initWithTarget:target];
}

- (id)forwardingTargetForSelector:(SEL)selector {
    return _target;
}

- (void)forwardInvocation:(NSInvocation *)invocation {
    void *null = NULL;
    [invocation setReturnValue:&null];
}

- (NSMethodSignature *)methodSignatureForSelector:(SEL)selector {
    return [NSObject instanceMethodSignatureForSelector:@selector(init)];
}

- (BOOL)respondsToSelector:(SEL)aSelector {
    return [_target respondsToSelector:aSelector];
}

- (BOOL)isEqual:(id)object {
    return [_target isEqual:object];
}

- (NSUInteger)hash {
    return [_target hash];
}

- (Class)superclass {
    return [_target superclass];
}

- (Class)class {
    return [_target class];
}

- (BOOL)isKindOfClass:(Class)aClass {
    return [_target isKindOfClass:aClass];
}

- (BOOL)isMemberOfClass:(Class)aClass {
    return [_target isMemberOfClass:aClass];
}

- (BOOL)conformsToProtocol:(Protocol *)aProtocol {
    return [_target conformsToProtocol:aProtocol];
}

- (BOOL)isProxy {
    return YES;
}

- (NSString *)description {
    return [_target description];
}

- (NSString *)debugDescription {
    return [_target debugDescription];
}

@end

这个时候,我们的timer的target要改一下,改成weakproxy:

self.timer = [NSTimer timerWithTimeInterval:1 target:[WeakProxy proxyWithTarget:self] selector:@selector(timerSchedue:) userInfo:nil repeats:YES];

我们运行来看看效果:


image.png

果然,delloc的方法执行了,说明,循环引用被打破了。

所以,后面如果用到NSTimer的时候,记得要注意这个问题哦.....

上一篇 下一篇

猜你喜欢

热点阅读