iOS之NSTimer导致的内存泄露

背景

公司的一个UIWebView要加个进度条,但是UIWebView没有进度啊。所以做个假的,用定时器。但是发现来回的进UIWebView的控制器,会崩溃。发现是在NSTimer那里崩溃的。那是因为NSTimer导致内存泄漏了。先说说为什么会发生内存泄漏吧。

写个小栗子。

//
//  ViewController.m
//  timer
//
//  Created by wyb on 2017/4/12.
//  Copyright © 2017年 中天易观. All rights reserved.
//

#import "ViewController.h"
#import "TimeViewController.h"

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
}


- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event
{
    TimeViewController *vc = [[TimeViewController alloc]init];
    [self presentViewController:vc animated:YES completion:nil];
}

@end

#import "TimeViewController.h"

@interface TimeViewController ()

@property(nonatomic,weak)NSTimer *timer;

@end

@implementation TimeViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    self.view.backgroundColor = [UIColor greenColor];
    
    self.timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(timeAction:) userInfo:nil repeats:YES];
    
    
}

- (void)timeAction:(NSTimer *)timer {
    
    NSLog(@"%s",__func__);
    
}

- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event
{
    [self dismissViewControllerAnimated:YES completion:nil];
}

发现个问题,当我跳到TimeViewController控制器的时候,定时器干活了。当我点击屏幕开始dismissViewController它的时候,发现定时器还在干活。那就是timer没停止。

delloc方法里销毁它。

- (void)dealloc
{
    [self.timer invalidate];
    self.timer = nil;
}

发现不起作用。我就是遇到这个问题导致的内存泄漏。本以为把它销毁了。然而并没有卵用。dealloc根本没走。

根源

@property(nonatomic,weak)NSTimer *timer;

self.timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(timeAction:) userInfo:nil repeats:YES];

timer创建成功会被添加到Runloop里,它就被Runloop强引用了。它的target是self。所以它强引用了self。属性那里是weak没事。
当我们想销毁TimeViewController的时候,因为它被timer强引用所以它销毁不了。多会timer死了,它才能死。但是它要是死不了delloc方法不会走。delloc方法走不了timer就死不了。只有执行了[self.timer invalidate],timer才会死。所以这就是循环引用了。

解决1

  __weak typeof(self) weakSelf = self;
    self.timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:weakSelf selector:@selector(timeAction:) userInfo:nil repeats:YES];

发现不起作用。还是引用self。这个里面表面看着可以,但是内部肯定还强引用。我这里也不太明白,知道的告诉下。

解决2

既然self被引用。我可以换个别的对象啊。让timer调用别的对象的timeAction,那么跟self就没半毛钱关系了。循环引用就迎刃而解了。写了个YBTimer。上代码吧

//
//  YBTimer.h
//  xxx
//
//  Created by wyb on 2017/4/11.
//  Copyright © 2017年 xxx. All rights reserved.
//

#import 

typedef void (^YBTimerBlock)(id userInfo);

@interface YBTimer : NSObject

+ (NSTimer *) scheduledTimerWithTimeInterval:(NSTimeInterval)interval
                                      target:(id)aTarget
                                    selector:(SEL)aSelector
                                    userInfo:(id)userInfo
                                     repeats:(BOOL)repeats;

+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)interval
                                      block:(YBTimerBlock)block
                                   userInfo:(id)userInfo
                                    repeats:(BOOL)repeats;

@end

//
//  YBTimer.m
//  xxx
//
//  Created by wyb on 2017/4/11.
//  Copyright © 2017年 xxx. All rights reserved.
//

#import "YBTimer.h"

//------------------------------YBTimerTargetBegin-------------------------------------


@interface YBTimerTarget : NSObject

@property (nonatomic, weak) id target;
@property (nonatomic, assign) SEL selector;
@property (nonatomic, weak) NSTimer* timer;

@end

@implementation YBTimerTarget

- (void)timeAction:(NSTimer *)timer {
    if(self.target) {

        [self.target performSelector:self.selector withObject:timer.userInfo afterDelay:0.0f];

    } else {
        [self.timer invalidate];
    }
}

@end


//------------------------------YBTimerTargetEnd-------------------------------------

@implementation YBTimer

+ (NSTimer *) scheduledTimerWithTimeInterval:(NSTimeInterval)interval
                                      target:(id)aTarget
                                    selector:(SEL)aSelector
                                    userInfo:(id)userInfo
                                     repeats:(BOOL)repeats {
    YBTimerTarget* timerTarget = [[YBTimerTarget alloc] init];
    timerTarget.target = aTarget;
    timerTarget.selector = aSelector;
    timerTarget.timer = [NSTimer scheduledTimerWithTimeInterval:interval
                                                         target:timerTarget
                                                       selector:@selector(timeAction:)
                                                       userInfo:userInfo
                                                        repeats:repeats];
    return timerTarget.timer;
}

+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)interval
                                      block:(YBTimerBlock)block
                                   userInfo:(id)userInfo
                                    repeats:(BOOL)repeats {
    NSMutableArray *userInfoArray = [NSMutableArray arrayWithObject:[block copy]];
    if (userInfo != nil) {
        [userInfoArray addObject:userInfo];
    }
    return [self scheduledTimerWithTimeInterval:interval
                                         target:self
                                       selector:@selector(timerBlock:)
                                       userInfo:[userInfoArray copy]
                                        repeats:repeats];
}

+ (void)timerBlock:(NSArray*)userInfo {
    YBTimerBlock block = userInfo[0];
    id info = nil;
    if (userInfo.count == 2) {
        info = userInfo[1];
    }
    
    if (block) {
        block(info);
    }
}

@end

NSTimer 进阶 http://www.jianshu.com/p/19aab8570ce3

你可能感兴趣的:(iOS之NSTimer导致的内存泄露)