Timer that updates label running in background

2020-04-17 03:38发布

I have coded a timer using an NSTimer that updates a label. The problem is that in the same viewcontroller I have a uitableview and when I scroll it down, the timer doesn't update its value so the user can "cheat" to stop the timer.

I think this could be easy to fix with a serial queue with CGD but I don't figure out how to do it.

Thanks in advance,

Juan

4条回答
我欲成王,谁敢阻挡
2楼-- · 2020-04-17 04:18

First of all keep in mind that you cannot perform UI changes in any other thread than the main thread. Having said that, you need the NSTimer to fire in the main queue, otherwise the program will crash when changing the UILabel. Have a look at this links http://bynomial.com/blog/?p=67 and this one http://developer.apple.com/library/mac/#documentation/Cocoa/Reference/Foundation/Classes/NSRunLoop_Class/Reference/Reference.html

To my knowledge, if you schedule the timer in the for the NSRunLoopCommonModes it will ignore event updates and fire the timer jsut how you want it:

 NSTimer *timer = [NSTimer timerWithTimeInterval:1.0
                                           target:self
                                         selector:@selector(timerDidTick:)
                                         userInfo:nil repeats:YES];
  [[NSRunLoop mainRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];

-(void) timerDidTick:(NSTimer*) theTimer{
    [[self myLabel] setText:@"Timer ticked!"];
}
查看更多
一纸荒年 Trace。
3楼-- · 2020-04-17 04:29

I have faced same problem if u run timer in main thread than when you scroll tableview its stop timer too. Solution is that you run timer in background and update GUI in main thread

    UIApplication *app = [UIApplication sharedApplication];
__block UIBackgroundTaskIdentifier bgTask = [app beginBackgroundTaskWithExpirationHandler:^{
    [app endBackgroundTask:bgTask];
    bgTask = UIBackgroundTaskInvalid;
}];

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
    //run function methodRunAfterBackground
    updateTimer1=[NSTimer scheduledTimerWithTimeInterval:0.1 target:self selector:@selector(updateGUIAudioPlayer) userInfo:nil repeats:YES];
    [[NSRunLoop currentRunLoop] addTimer:updateTimer1 forMode:NSDefaultRunLoopMode];
    [[NSRunLoop currentRunLoop] run];
});

    -(void)updateGUIAudioPlayer {
dispatch_async(dispatch_get_main_queue(), ^{
    self.label.text = @"your text";
});

    //but if you want to stop timer by himself than use main thread too like that

    dispatch_async(dispatch_get_main_queue(), ^{
             [updateTimer1 invalidate];
         });}
查看更多
Summer. ? 凉城
4楼-- · 2020-04-17 04:33

To update the UI you need to use OperationQueue.main. Supposing you have a UILabelnamed timeLabel:

var timeInSeconds = 0

timer = Timer(timeInterval: 1, repeats: true, block: { timer in
        timeInSeconds += 1
        OperationQueue.main.addOperation {
            self.timeLabel.text = String(timeInSeconds)
        }
    })

RunLoop.main.add(timer, forMode: .commonModes)
查看更多
对你真心纯属浪费
5楼-- · 2020-04-17 04:40
-(void) timerDidTick:(NSTimer*) theTimer{
        _currentNumber =_currentNumber+1;
        //check if current number is bigger than the contents of the word array
        if (_currentNumber <=wordArray.count-1) {
            NSLog(@"_currentNumber @%i wordArray @%i",_currentNumber,wordArray.count);
            _RandomLoadingText.text = @"Timer" ;
            [self changeTextInRandomLoadingLabel:_currentNumber];
        }

        else if (_currentNumber >=wordArray.count-1)
         {
             NSLog(@"reached end of array");
             [timer invalidate];
         }
    }

-(void)changeTextInRandomLoadingLabel:(int)myInt
{
    _RandomLoadingText.text = [wordArray objectAtIndex:myInt];
}

//--------- view did load

    _RandomLoadingText.text = @"Test";
    _currentNumber =0;
    timer = [NSTimer timerWithTimeInterval:1.0
                                             target:self
                                           selector:@selector(timerDidTick:)
                                           userInfo:nil repeats:YES];
    [[NSRunLoop mainRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
查看更多
登录 后发表回答