SoFunction
Updated on 2025-04-03

The most complete tutorial on using timers in iOS

Preface

I believe that when it comes to timers, the ones we use the most are NSTimer and GCD, and there is another advanced timer, CADisplayLink;. The following will introduce you in detail about the use of iOS timers. Without further ado, let’s take a look at the detailed introduction.

1. NSTimer

There are several initialization methods for NSTimer:

It will automatically start and be added to the NSDefaultRunLoopMode of MainRunloop,

Notice:The automatic startup here does not start immediately, but will delay about an interval:

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

parameter:

  • internal: time interval, how often do I call it
  • repeats: whether to call repeatedly
  • block: Things that need to be repeated

use:

 [NSTimer scheduledTimerWithTimeInterval:1 repeats:YES block:^(NSTimer * _Nonnull timer) {
 
 static NSInteger num = 0;
 
 NSLog(@"%ld", (long)num);
 num++;
 
 if (num > 4) {
 
 [timer invalidate];
 
 NSLog(@"end");
 }
 }];
 
 NSLog(@"start");

At this time, the console output:

2016-12-29 16:29:53.901 Timer[11673:278678] start
2016-12-29 16:29:54.919 Timer[11673:278678] 0
2016-12-29 16:29:55.965 Timer[11673:278678] 1
2016-12-29 16:29:56.901 Timer[11673:278678] 2
2016-12-29 16:29:57.974 Timer[11673:278678] 3
2016-12-29 16:29:58.958 Timer[11673:278678] 4
2016-12-29 16:29:58.959 Timer[11673:278678] end

It can be seen that the internal here is set to 1s, and the content in the block is delayed by about 1s before it starts to execute;

I did the stop timer here directly in the block. If I use a global variable to manually stop the timer in another place, I need to do this:

[ invalidate];
 = nil;
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti invocation:(NSInvocation *)invocation repeats:(BOOL)yesOrNo

parameter:

  • ti: Repeat execution time interval
  • invocation: NSInvocation instance, see the usage of itBasic usage of NSInvocation
  • yesOrNo: Whether to execute repeatedly

Example:

// NSInvocation form- (void)timer2 {
 
 NSMethodSignature *method = [ViewController instanceMethodSignatureForSelector:@selector(invocationTimeRun:)];
 
 NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:method];
 NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:1.0 invocation:invocation repeats:YES];
 
 // Set method caller  = self;
 
 // The SEL here needs to be consistent with the NSMethodSignature  = @selector(invocationTimeRun:);
 
 // Set parameters // //The Index here starts from 2, thinking that 0 and 1 have been occupied, namely self (target) and selector (_cmd) // If there are multiple parameters, you can set 3 4 5 ... [invocation setArgument:&timer atIndex:2];
 
 [invocation invoke];
 
 NSLog(@"start");
}
- (void)invocationTimeRun:(NSTimer *)timer {
 
 static NSInteger num = 0;
 NSLog(@"%ld---%@", (long)num, timer);
 
 num++;
 
 if (num > 4) {
 [timer invalidate];
 }
}

Output:

2016-12-29 16:52:54.029 Timer[12089:289673] 0---<__NSCFTimer: 0x60000017d940>
2016-12-29 16:52:54.029 Timer[12089:289673] start
2016-12-29 16:52:55.104 Timer[12089:289673] 1---<__NSCFTimer: 0x60000017d940>
2016-12-29 16:52:56.095 Timer[12089:289673] 2---<__NSCFTimer: 0x60000017d940>
2016-12-29 16:52:57.098 Timer[12089:289673] 3---<__NSCFTimer: 0x60000017d940>
2016-12-29 16:52:58.094 Timer[12089:289673] 4---<__NSCFTimer: 0x60000017d940>

It can be seen that the timer is executed immediately, without delay;

This method can pass multiple parameters. Here is an example of passing two parameters:

// NSInvocation form- (void)timer2 {
 
 NSMethodSignature *method = [ViewController instanceMethodSignatureForSelector:@selector(invocationTimeRun:des:)];
 
 NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:method];
 NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:1.0 invocation:invocation repeats:YES];
 
 // Set method caller  = self;
 
 // The SEL here needs to be consistent with the NSMethodSignature  = @selector(invocationTimeRun:des:);
 
 // Set parameters // //The Index here starts from 2, thinking that 0 and 1 have been occupied, namely self (target) and selector (_cmd) // If there are multiple parameters, you can set 3 4 5 ... [invocation setArgument:&timer atIndex:2];
 // Set the second parameter NSString *dsc = @"The second parameter is a string";
 [invocation setArgument:&dsc atIndex:3];
 
 [invocation invoke];
 
 NSLog(@"start");
}
- (void)invocationTimeRun:(NSTimer *)timer des:(NSString *)dsc {
 
 static NSInteger num = 0;
 NSLog(@"%ld---%@--%@", (long)num, timer, dsc);
 
 num++;
 
 if (num > 4) {
 [timer invalidate];
 }
}

Output:

2016-12-29 16:57:45.087 Timer[12183:292324] 0---<__NSCFTimer: 0x60000016dbc0>--The second parameter is a string
2016-12-29 16:57:45.088 Timer[12183:292324] start
2016-12-29 16:57:46.161 Timer[12183:292324] 1---<__NSCFTimer: 0x60000016dbc0>--The second parameter is a string
2016-12-29 16:57:47.161 Timer[12183:292324] 2---<__NSCFTimer: 0x60000016dbc0>--The second parameter is a string
2016-12-29 16:57:48.150 Timer[12183:292324] 3---<__NSCFTimer: 0x60000016dbc0>--The second parameter is a string
2016-12-29 16:57:49.159 Timer[12183:292324] 4---<__NSCFTimer: 0x60000016dbc0>--The second parameter is a string
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo

parameter:

  • ti: Time interval
  • aTarget: caller
  • aSelector: Method of execution
  • userInfo: Parameters
  • yesOrNo: Whether to execute repeatedly

Example:

- (void)timer3 {
 
 NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:1 target:self selector:@selector(targetRun:) userInfo:@"This is the parameter carried" repeats:YES];
 
 NSLog(@"start");
}
- (void)targetRun:(NSTimer *)timer {
 
 static NSInteger num = 0;
 
 NSLog(@"%ld---%@--%@", (long)num, timer, );
 
 num++;
 
 if (num > 4) {
 [timer invalidate];
 }
}

Output:

2016-12-29 17:05:11.590 Timer[12328:296879] start
2016-12-29 17:05:12.655 Timer[12328:296879] 0---<__NSCFTimer: 0x608000162700>--This is the parameter carried
2016-12-29 17:05:13.661 Timer[12328:296879] 1---<__NSCFTimer: 0x608000162700>--This is the parameter carried
2016-12-29 17:05:14.664 Timer[12328:296879] 2---<__NSCFTimer: 0x608000162700>--This is the parameter carried
2016-12-29 17:05:15.651 Timer[12328:296879] 3---<__NSCFTimer: 0x608000162700>--This is the parameter carried
2016-12-29 17:05:16.650 Timer[12328:296879] 4---<__NSCFTimer: 0x608000162700>--This is the parameter carried

The usage of creating timers in the following three ways is similar to the corresponding methods above. It should be noted that the timer created in this way will not be executed, and we need to manually turn on the timer;

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

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

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

The way to enable it is to add the current timer to RunLoop:

[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];

Here is an example:

- (void)timer4 {
 
 NSTimer *timer = [NSTimer timerWithTimeInterval:1 repeats:YES block:^(NSTimer * _Nonnull timer) {
 
 static NSInteger num = 0;
 
 NSLog(@"%ld", (long)num);
 num++;
 
 if (num > 4) {
 
 [timer invalidate];
 timer = nil;
 
 NSLog(@"end");
 }
 }];
 
 [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];
 
 NSLog(@"start");
}

Output:

2016-12-29 17:12:13.955 Timer[12498:301751] start
2016-12-29 17:12:15.013 Timer[12498:301751] 0
2016-12-29 17:12:16.018 Timer[12498:301751] 1
2016-12-29 17:12:17.011 Timer[12498:301751] 2
2016-12-29 17:12:18.024 Timer[12498:301751] 3
2016-12-29 17:12:19.023 Timer[12498:301751] 4
2016-12-29 17:12:19.023 Timer[12498:301751] end

This is all the basic way to create a timer. You can also set other properties, such as the turn-on time, which can be set directly with its API;

Notice:In the above example, I did not use the global NSTimer object. If you set the global variable, or set it to a property, you should manually set it to nil when stopping the timer, that is:

[timer invalidate];
 timer = nil;

II. GCD

dispatch_after: delay execution once

dispatch_after(dispatch_time_t when, dispatch_queue_t queue, dispatch_block_t block)

Example:

- (void)gcdTimer {
 
 // Delay 2s dispatch_time_t delayTime = dispatch_time(DISPATCH_TIME_NOW, 2 * NSEC_PER_SEC);
 
 dispatch_after(delayTime, dispatch_get_main_queue(), ^(void){
 
 NSLog(@"Execute after 2s delay");
 });
 
 NSLog(@"start");
}

Repeated execution timer

void
dispatch_source_set_timer(dispatch_source_t source,
 dispatch_time_t start,
 uint64_t interval,
 uint64_t leeway)

parameter:

  • source: timer
  • start: start time, when we use dispatch_time or DISPATCH_TIME_NOW, the system will use the default clock to time. However, when the system is sleeping, the default clock does not go, which will cause the timer to stop. Use dispatch_walltime to allow the timer to time at real time intervals;
  • interval: interval (if set to DISPATCH_TIME_FOREVER, it will only be executed once)
  • leeway: The allowable error range; the timing cannot be 100% accurate. Even if it is set to 0, it is not 100% accurate, so a reasonable allowable error can be set, unit: nanoseconds (NSEC_PER_SEC)

For related content, please refer to the article:Use of Dispatch Source Timer and precautions

// Repeated execution timer- (void)gcdTimer1 {
 
 // Get the global queue dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
 // Create a timer dispatch_source_t _timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue);
 
 // Start time dispatch_time_t start = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1.0 * NSEC_PER_SEC));
 
// dispatch_time_t start = dispatch_walltime(NULL, 0);
 
 // Repeat interval uint64_t interval = (uint64_t)(1.0 * NSEC_PER_SEC);
 
 // Set the timer dispatch_source_set_timer(_timer, start, interval, 0);
 
 // Set the event to be executed dispatch_source_set_event_handler(_timer, ^{
 
 //Execute events here static NSInteger num = 0;
 
 NSLog(@"%ld", (long)num);
 num++;
 
 if (num > 4) {
 
 NSLog(@"end");
 
 // Turn off the timer dispatch_source_cancel(_timer);
 }
 });
 // Turn on the timer dispatch_resume(_timer);
 
 NSLog(@"start");
}

Output:

2016-12-30 10:15:01.114 Timer[3393:99474] start
2016-12-30 10:15:02.187 Timer[3393:99796] 0
2016-12-30 10:15:03.114 Timer[3393:99796] 1
2016-12-30 10:15:04.186 Timer[3393:99796] 2
2016-12-30 10:15:05.188 Timer[3393:99796] 3
2016-12-30 10:15:06.188 Timer[3393:99796] 4
2016-12-30 10:15:06.188 Timer[3393:99796] end

The start time here is set to an interval of 1s, so execution is only started after 1s. You can set the DISPATCH_TIME_NOW to execute immediately;

Notice:

The start time here can be set in the following way:

dispatch_time_t start = dispatch_walltime(NULL, 0);

Or set directly to: DISPATCH_TIME_NOW

Regarding the difference between dispatch_walltime and dispatch_time, it is also mentioned above, and you can also refer to it.*The answer above; the main difference is that the former will continue to time when the system is sleeping, while the latter will stop time when the system is sleeping, and will continue to time when the system is reactivated;

Stop timer:

How to stop the GCD timer,Use of Dispatch Source Timer and precautionsThere are mainly two types of mentions:

// Turn off the timer// Completely destroy the timer, if it is re-activated, it needs to be re-created// Global variable, it needs to be set to nil after closingdispatch_source_cancel(_timer);
 
// Pause the timer// You can use dispatch_resume(_timer) to turn on again// Global variables cannot be set to nil after pause, otherwise they cannot be restarteddispatch_suspend(_timer);

3. CADisplayLink

CADisplayLink runs 60 times per second by default. The frameInterval property changes the number of frames per second. If set to 2, it means that CADisplayLink runs every other frame, and the effective logic runs 30 times per second.

Called on screen refresh: CADisplayLink is a timer class that allows us to draw specific content on the screen at a frequency synchronized with the screen refresh rate. After CADisplayLink registers to runloop in a specific mode, whenever the screen display content refresh ends, runloop will send a specified selector message to the target specified by CADisplayLink, and the selector corresponding to the CADisplayLink class will be called once. So usually, according to the refresh rate of iOS device screen 60 times/second

Delay: The screen refresh frequency of iOS devices is fixed. Under normal circumstances, CADisplayLink will be called at the end of each refresh, with very high accuracy. However, if the method called is time-consuming and exceeds the screen refresh cycle, several callback calls will be skipped.

If the CPU is too busy and cannot guarantee the refresh rate of 60 times per second on the screen, it will lead to several chances of calling the callback method, and the number of skipping depends on the busyness of the CPU.

Usage scenario: From the principle, it can be seen that CADisplayLink is suitable for continuous repainting of the interface. For example, when playing video, you need to continuously obtain the next frame for interface rendering.

+ (CADisplayLink *)displayLinkWithTarget:(id)target selector:(SEL)sel

parameter:

  • target: caller
  • sel: method to execute

Example:

- (void) displayLink {
 
 CADisplayLink *display = [CADisplayLink displayLinkWithTarget:self selector:@selector(displayRun:)];
 
 // Execute it once every 1s// Value range is 1--100. The larger the value, the higher the frequency  = 2;
 
 [display addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
}
- (void)displayRun:(CADisplayLink *)link {
 
 static NSInteger num = 0;
 
 NSLog(@"%ld", (long)num);
 num++;
 
 if (num > 4) {
 
 [link invalidate];
 
 NSLog(@"end");
 }
}

The example here is not appropriate and should not be used in this case.

In addition, we can use its paused property to pause it, or continue:

// pause  = YES;
// continue  = NO;

Summarize

The above is the entire content of this article. I hope that the content of this article has certain reference value for everyone's study or work. If you have any questions, you can leave a message to communicate. Thank you for your support.