美文网首页
解决定时器循环引用的问题

解决定时器循环引用的问题

作者: ZZ_军哥 | 来源:发表于2022-03-08 14:35 被阅读0次

关于NSTimer和CADisplayLink定时器循环引用;
我这里有控制器ViewControll1和ViewControll2两个控制器, ViewControll1 push出来ViewControll2,然后在ViewControll2写下这些代码,对button,NSTimer,和CADisplayLink进行验证,其中TestBtn继承自UIButton,重写dealloc方法,观察能否销毁

@interface ViewController2 ()

@property(nonatomic,strong)TestBtn *btn;

@property(nonatomic,strong)NSTimer *timer;

@property(nonatomic,strong)CADisplayLink *displayLink;

@end

@implementation ViewController2

- (void)viewDidLoad {
    [super viewDidLoad];
    self.view.backgroundColor = [UIColor greenColor];
    
    self.btn = [[TestBtn alloc]initWithFrame:CGRectMake(0, 0, 100, 50)];
    self.btn.center = self.view.center;
    [self.btn setBackgroundColor:[UIColor blueColor]];
    [self.btn setTitle:@"按钮" forState:UIControlStateNormal];
    [self.btn addTarget:self action:@selector(action) forControlEvents:UIControlEventTouchUpInside];
    [self.view addSubview:self.btn];
    
    __weak typeof(self) weakSelf = self;
//    self.timer = [NSTimer timerWithTimeInterval:1 repeats:YES block:^(NSTimer * _Nonnull timer) {
//        [weakSelf timerAction];
//    }];
    
    self.timer =  [NSTimer scheduledTimerWithTimeInterval:1 target:weakSelf selector:@selector(timerAction) userInfo:nil repeats:YES];
    
    
    [[NSRunLoop mainRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];
//    self.displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(displayLinkAction)];
//    [self.displayLink addToRunLoop:[NSRunLoop mainRunLoop] forMode:NSRunLoopCommonModes];
}

- (void)action
{
    NSLog(@"执行点击");
}
- (void)timerAction
{
    NSLog(@"执行timer定时器");
}
- (void)displayLinkAction
{
    NSLog(@"执行displayLink定时器");
}

- (void)dealloc
{
  [self.timer invalidate];
    NSLog(@"%s",__func__);
}

1.只打开TestBtn的方法,运行进入VC2,再POP回VC1,打印内容


image.png

代表button和VC2没有造成循环引用

2.只打开block类型的timer代码,并重新进行上面的操作

    self.timer = [NSTimer timerWithTimeInterval:1 repeats:YES block:^(NSTimer * _Nonnull timer) {
        [self timerAction];
    }];
    [[NSRunLoop mainRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];
image.png

回到VC1时,timer还在执行,并且VC2也没销毁,造成了循环引用
VC2->timer->timer的block对[self timerAction]变量捕捉VC2,造成了循环引用,可以通过__weak对[self timerAction]的捕捉VC2为弱引用

    __weak typeof(self) weakSelf = self;
    self.timer = [NSTimer timerWithTimeInterval:1 repeats:YES block:^(NSTimer * _Nonnull timer) {
        [weakSelf timerAction];
    }];
    [[NSRunLoop mainRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];

打印信息,显示这样就不会造成循环 引用


image.png

但是有其他情况,比如不是通过block捕捉的self,而是通过target引用的,那么__weak就用不了了,比如这样,我试过了VC2不会销毁

    __weak typeof(self) weakSelf = self;
    self.timer =  [NSTimer scheduledTimerWithTimeInterval:1 target:weakSelf selector:@selector(timerAction) userInfo:nil repeats:YES];
    [[NSRunLoop mainRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];

这种可以通过新建一个类,然后VC2->强timer->新建类强引用->弱VC2的方式进行解决,我这里新建了一个类TestProxy

@interface TestProxy : NSObject

@property(nonatomic,weak)id target;

+ (instancetype)proxyWithTarget:(id)target;

@end
@implementation TestProxy

+ (instancetype)proxyWithTarget:(id)target
{
    TestProxy *proxy = [[TestProxy alloc] init];
    proxy.target = target;
    return proxy;
}

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

然后修改self.timer的创建方式

    self.timer =  [NSTimer scheduledTimerWithTimeInterval:1 target:[TestProxy proxyWithTarget:self] selector:@selector(timerAction) userInfo:nil repeats:YES];
    [[NSRunLoop mainRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];

打印结果为,证明也是可以销毁的


image.png

但是上面的内容可以再优化,由于TestProxy继承自NSObject,它需要经过消息发送的遍历查找,动态解析,最后才会到我们这里的消息转发过程,大大消耗性能;苹果提供了一个专门做消息转发的类NSProxy,这个类和NSObject同一级别,该类的特性就是去掉消息遍历查找,消息添加,直接消息转发的过程,大大提高了效率

@interface GYJProxy : NSProxy

+ (instancetype)proxyWithTarget:(id)target;

@end
@interface GYJProxy ()

@property(nonatomic,weak)id target;

@end
@implementation GYJProxy

+ (instancetype)proxyWithTarget:(id)target
{
    GYJProxy *proxy = [GYJProxy alloc];
    proxy->_target = target;
    return proxy;
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)sel
{
    return [_target methodSignatureForSelector:sel];
}
- (void)forwardInvocation:(NSInvocation *)invocation
{
    [invocation invokeWithTarget:_target];
}
@end

最后验证的结果也是可以正常销毁的,CADisplayLink和NSTimer一样,也会造成循环引用,处理方式和NSTimer一样.还有就是NSTimer和CADisplayLink都是基于runloop使用的,这会造成时间稍微不准,建议采用GCD的定时器,GCD的定时器不依赖runloop,时间是准的

    dispatch_source_t timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, dispatch_get_main_queue());
    //1.源 2.开始时间 3.间隔时间 4.精准度
    dispatch_source_set_timer(timer, DISPATCH_TIME_NOW, NSEC_PER_SEC*1.0, 0*NSEC_PER_SEC);
    dispatch_source_set_event_handler(timer, ^{
        NSLog(@"定时器执行的任务");
    });
    dispatch_resume(timer);

相关文章

网友评论

      本文标题:解决定时器循环引用的问题

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