美文网首页
NSTimer使用

NSTimer使用

作者: 云天涯丶 | 来源:发表于2018-04-10 13:40 被阅读34次

一、介绍

NSTimer.h文件里的一些方法、属性的讲解

1、NSInvocation创建

+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)ti invocation:(NSInvocation *)invocation repeats:(BOOL)yesOrNo;
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti invocation:(NSInvocation *)invocation repeats:(BOOL)yesOrNo;

需要一个NSInvocation对象创建。前者创建一个定时器,但是需要在创建定时器后手动将timer加入NSRunLoop 中;后者自动以NSDefaultRunLoopMode添加到当前线程RunLoop中。

例子:

- (void)setupTimer{
    SEL sel = @selector(run);
    NSMethodSignature *sign = [[self class] instanceMethodSignatureForSelector:sel];
    NSInvocation *invo = [NSInvocation invocationWithMethodSignature:sign];
    invo.target = self;
    invo.selector = sel;
    [invo invoke];
    
    NSTimer *timer = [NSTimer timerWithTimeInterval:1 invocation:invo repeats:YES];
    // NSRunLoopCommonModes NSDefaultRunLoopMode
    [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];
    
    
}

- (void)run{
    NSLog(@"run");
}

2、指定target和selector

+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;

这两个是常用的创建方式,区别同上

3、block

+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block ;
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block;

block创建方式,区别同上

4、实例方法

- (instancetype)initWithFireDate:(NSDate *)date interval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block ;
- (instancetype)initWithFireDate:(NSDate *)date interval:(NSTimeInterval)ti target:(id)t selector:(SEL)s userInfo:(nullable id)ui repeats:(BOOL)rep ;

实例方法创建方式,区别同上

// 执行(不加入runloop,用fire执行一次)
- (void)fire;

@property (copy) NSDate *fireDate;
@property (readonly) NSTimeInterval timeInterval;

// 容差  
@property NSTimeInterval tolerance API_AVAILABLE(macos(10.9), ios(7.0), watchos(2.0), tvos(9.0));
// 停止定时器再次触发并请求将其从runloop中移除
- (void)invalidate;
@property (readonly, getter=isValid) BOOL valid;

@property (nullable, readonly, retain) id userInfo;

二、实例

1、子线程开启定时器

@interface BViewController ()

@property (nonatomic,strong) NSTimer *timer;
@property (nonatomic,strong) NSThread *thread;

@end

@implementation BViewController

- (void)dealloc{
    NSLog(@"注销");
}

- (void)viewDidLoad {
    [super viewDidLoad];
    NSLog(@"%@ %@",[NSThread currentThread],[NSRunLoop currentRunLoop]);
    self.thread = [[NSThread alloc] initWithTarget:self selector:@selector(perTask) object:nil];
    [self.thread start];
    
}

- (void)perTask{
    NSLog(@"%@ %@",[NSThread currentThread],[NSRunLoop currentRunLoop]);
    self.timer = [NSTimer scheduledTimerWithTimeInterval:1 target:self selector:@selector(run) userInfo:nil repeats:YES];
    // 子线程runloop是默认不运行,必须手动开启
    [[NSRunLoop currentRunLoop] run];
}

- (void)run{
    NSLog(@"run里的线程%@",[NSThread currentThread]);
    if ([NSThread currentThread].isCancelled) {
        [self.timer invalidate];
    }
    NSLog(@"run");
}

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
    [self.thread cancel];
//    if (self.thread.isCancelled) {
//        [self.timer invalidate];
//    }
    NSLog(@"touch的线程%@",[NSThread currentThread]);
    
}

这个例子有几个注意点:
1、子线程runloop默认不运行,必须手动run
2、invalidate:文档上有一段说明

必须从定时器当前线程发送invalidate。 如果从另一个线程发送此消息,则与定时器关联的输入源可能不会从其runloop中删除,这可能会阻止线程正常退出。

所以销毁timer必须在子线程中即run方法里

2、循环问题
这个NStimer经典问题:当前VC持有timer,timer的target又是self(VC),从而产生循环引用。

怎么解决?

一般是按实例1中的方法,在退出VC前invalidate掉timer,但是往往我们会遗漏或者某些场景不好操作。如何优雅的解决循环引用呢?

把timer的target设为自己,这样就不会循环引用。我们用一个Category封装一下:

+ (NSTimer *)ca_scheduledTimerWithTimeInterval:(NSTimeInterval)interval
                                       repeats:(BOOL)repeats
                                         block:(void(^)(void))block{
    return [self scheduledTimerWithTimeInterval:interval target:self selector:@selector(ca_blockInvoke:) userInfo:[block copy] repeats:repeats];
}

+ (void)ca_blockInvoke:(NSTimer *)timer{
    void (^block)(void) = timer.userInfo;
    if(block) {
        block();
    }
}

这样的话,timer事件在block中执行。

注:
1、NSTimer不是一种实时机制,这点官网文档上作了说明

A timer is not a real-time mechanism. If a timer’s firing time occurs during a long run loop callout or while the run loop is in a mode that isn't monitoring the timer, the timer doesn't fire until the next time the run loop checks the timer. Therefore, the actual time at which a timer fires can be significantly later.

2、当然了,想让NSTimer在scrollView滚动时有效需要将runloop mode换成NSRunLoopCommonModes(默认mode是NSDefaultRunLoopMode,scrollView滚动时mode切换成UITrackingRunLoopMode,所以NSTimer失效)

相关文章

  • NSTimer的循环引用

    NSTimer基本使用 NSTimer与RunLoop NSTimer 循环引用的问题 如何在子线程使用NSTim...

  • iOS-NSTimer-循环引用问题

    在使用NSTimer的时候,NSTimer会生成指向其使用者的引用,而其使用者如果也引用了NSTimer,那么就会...

  • NSTimer的使用

    NSTimer 的使用 为什么会写NSTimer呢? 原因很简单, 这里有坑! NSTimer 使用的顺序 创建N...

  • 内存管理总结

    CADisplayLink、NSTimer使用注意 CADisplayLink、NSTimer会对target产生...

  • Objective-C基础-内存管理

    1、CADisplayLink、NSTimer使用 CADisplayLink、NSTimer会对target产生...

  • CADisplayLink、NSTimer使用注意

    CADisplayLink、NSTimer使用注意 CADisplayLink、NSTimer会对target产生...

  • 底层-内存管理

    CADisplayLink、NSTimer使用注意 CADisplayLink、NSTimer会对target产生...

  • 内存管理

    CADisplayLink、NSTimer使用注意 CADisplayLink、NSTimer会对target产生...

  • iOS内存管理

    CADisplayLink、NSTimer使用注意 CADisplayLink、NSTimer会对target产生...

  • 内存管理

    CADisplayLink、NSTimer使用注意 CADisplayLink、NSTimer会对target产生...

网友评论

      本文标题:NSTimer使用

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