NSTimer

作者: 开洋_shen | 来源:发表于2018-11-10 17:25 被阅读0次

    scheduledTimerWith和timerWith和区别

    那每种方式的调用接口又分为scheduledTimerWith和timerWith是为什么呢?这是因为NSTimer是加到runloop中执行的。看scheduledTimerWith的函数说明,创建并安排到runloop的default mode中。

    使用 timerWithTimeInterval 加入runLoop 
    NSTimer *timer  =  [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(Timered) userInfo:nil repeats:YES];
    
    [[NSRunLoop mainRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];
    

    1. 子线程启动timer:

    iOS是通过runloop作为消息循环机制,主线程默认启动了runloop,可是子线程没有默认的runloop,因此,我们在子线程启动定时器是不生效的。主动去启动

    dispatch_async(dispatch_get_global_queue(0, 0), ^{
            NSTimer* timer = [NSTimer timerWithTimeInterval:1 target:self selector:@selector(Timered:) userInfo:nil repeats:YES];
            [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
            [[NSRunLoop currentRunLoop] run];
        });
    
    1. runLoop Mode:

    使用schedule 启动定时器是默认在runloop的NSDefaultRunLoopMode的,
    最常见的问题就是在UITrackingRunLoopMode,即UIScrollView滑动过程中定时器失效。
    解决方式就是把timer add到runloop的NSRunLoopCommonModes。

    1. 循环引用问题:

    就是NSTimer的target被强引用了,而target就是所在的控制器,他又强引用的timer,造成了循环引用。

    不是所有的NSTimer都会造成循环引用.
    
    • 非repeat类型的。非repeat类型的timer不会强引用target,因此不会出现循环引用。

    • block类型的,新api。iOS 10之后才支持,因此对于还要支持老版本的app来说,这个API暂时无法使用。当然,block内部的循环引用也要避免。

    解决了循环引用,target不一定就可以释放了,别忘了在持有timer的类的析构函数执行的时候执行invalidate。
    
    NSTimer* timer = [NSTimer scheduledTimerWithTimeInterval:1 target:self selector:@selector(Timered:) userInfo:nil repeats:YES];
    

    timer 没有被self引用,但是self还是不能被释放.因为timer默认被加到runloop中 timer强引用了self .self也不会释放;

    NSTimer会保留其目标对象
    

    一、NSTimer 分类实现block

    - (void)viewDidLoad {
        [super viewDidLoad];
        __weak id weakSelf = self;
        NSTimer* timer = [NSTimer scheduledTimerWithTimeInterval:1 repeats:YES block:^(NSTimer *timer) {
            NSLog(@"block %@",weakSelf);
        }];
    }
    
    @implementation NSTimer(BlockTimer)
    + (NSTimer*)scheduledTimerWithTimeInterval:(NSTimeInterval)interval repeats:(BOOL)repeats blockTimer:(void (^)(NSTimer *))block{
        NSTimer* timer = [NSTimer scheduledTimerWithTimeInterval:interval target:self selector:@selector(timered:) userInfo:[block copy] repeats:repeats];
        return timer;
    }
    
    + (void)timered:(NSTimer*)timer {
        void (^block)(NSTimer *timer)  = timer.userInfo;
        block(timer);
    }
    @end
    
    

    1.将强引用的target变成了NSTimer的类对象。类对象本身是单例的,是不会释放的,所以强引用也无所谓。
    2.执行的block通过userInfo传递给定时器的响应函数timered:。
    循环引用被打破的结果是:
    timer的使用者强引用timer。
    timer强引用NSTimer的类对象。
    timer的使用者在block中通过weak的形式使用,因此是被timer弱引用。

    二、NSProxy的方式

    建立一个proxy类,让timer强引用这个实例,这个类中对timer的使用者target采用弱引用的方式,再把需要执行的方法都转发给timer的使用者。

    @interface ProxyObject : NSProxy
    @property (weak, nonatomic) id target;
    + (instancetype)proxyWithTarget:(id)target;
    @end
    
    @implementation ProxyObject
    
    + (instancetype)proxyWithTarget:(id)target {
        ProxyObject* proxy = [[self class] alloc];
        proxy.target = target;
        return proxy;
    }
    
    - (NSMethodSignature *)methodSignatureForSelector:(SEL)sel{
        return [self.target methodSignatureForSelector:sel];
    }
    
    - (void)forwardInvocation:(NSInvocation *)invocation{
        SEL sel = [invocation selector];
        if ([self.target respondsToSelector:sel]) {
            [invocation invokeWithTarget:self.target];
        }
    }
    
    @end
    
    @implementation ProxyTimer
    + (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo{
        NSTimer* timer = [NSTimer scheduledTimerWithTimeInterval:ti target:[ProxyObject proxyWithTarget: aTarget] selector:aSelector userInfo:userInfo repeats:yesOrNo];
        return timer;
    }
    @end
    

    三、封装timer,弱引用target

    @interface NormalTimer : NSObject
    @property (nonatomic, weak) id target;
    @property (nonatomic) SEL selector;
    @end
    
    @implementation NormalTimer
    - (void)dealloc{
        NSLog(@"timer dealloc");
    }
    
    - (void)timered:(NSTimer*)timer{
        [self.target performSelector:self.selector withObject:timer];
    }
    @end
    
    @interface NSTimer(NormalTimer)
    + (NSTimer *)scheduledNormalTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;
    @end
      
    @implementation NSTimer(NormalTimer)
    + (NSTimer *)scheduledNormalTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo{
        NormalTimer* normalTimer = [[NormalTimer alloc] init];
        normalTimer.target = aTarget;
        normalTimer.selector = aSelector;
        NSTimer* timer = [NSTimer scheduledTimerWithTimeInterval:ti target:normalTimer selector:@selector(timered:) userInfo:userInfo repeats:yesOrNo];
        return timer;
    }
    @end
    
    

    相关文章

      网友评论

          本文标题:NSTimer

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