iOS中的网络和多线程编程(十)

简介: iOS中的网络和多线程编程(十)

iOS中如何触发定时任务或延时任务


定时任务指周期性地调用某个方法,实现任务的反复执行,如倒计时等;延时任务指等待一定的时间后再执行某个任务,如页面的延时跳转等。iOS中控制任务的延时或定时执行的方法有很多,使用中要注意是同步还是异步,是否会阻塞主线程等问题。延时和定时的实现方法依次如下。


1.performSelector实现延时任务


延时任务可以通过当前UIViewController的performSelector隐式创建子线程实现,不会阻塞主线程。


/*延迟10s执行任务*/
[self performSelector:@selector(task) withObject:nil afterDelay:10];
- (void)task
{
    //delay task
}


2.利用sleep实现后面任务的等待

慎用,会阻塞主线程NSThreadsleepForTimeInterval:10.0];


3.GCD实现延时或定时任务

通过GCD实现block代码块的延时执行。


dispatch_time_t delay = dispatch_time(DISPATCH_TIME_NOW, 10 * NSEC_PER_SEC);
dispatch_after(delay, dispatch_get_main_queue(), ^{
    //delay task
});


GCD还可以用来实现定时器功能,还能设置延时开启计时器,使用中注意一定要定义强引用指针来指向计时器对象才可让计时器生效。


/*必须要用强引用指针,计时器才会生效*/
@property (nonatomic, strong) dispatch_source_t timer;
/*在指定线程上定义计时器*/
dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
dispatch_source_t _timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue);
/*开始的时间*/
dispatch_time_t when = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1.0 * NSEC_PER_SEC));
/*设置计时器*/
dispatch_source_set_timer(_timer, when, 1.0 * NSEC_PER_SEC, 0);
/*计时器回调block*/
dispatch_source_set_event_handler(_timer, ^{
   NSLog(@"dispatch_source_set_timer is working!");
});
/*开启计时器*/
dispatch_resume(_timer);
/*强引用计时器对象*/
self.timer = _timer;


4.NSTimer实现定时任务


NSTimer主要用于开启定时任务,但要正确使用才能保证它能够正常有效地运行。尤其要注意以下两点:


1)确保NSTimer已经添加到当前RunLoop。

2)确保当前RunLoop已经启动。


创建NSTimer有两种方法,代码如下:


+ (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;


这两种方法的主要区别为,使用timerWithTimeInterval创建的timer不会自动添加到当前RunLoop中,需要手动添加并指定RunLoop的模式:[[NSRunLoop currentRunLoop] addTimer:mainThreadTimer forMode:NSDefaultRunLoopMode];而使用scheduledTimerWithTimeInterval创建的RunLoop会默认添加到当前RunLoop中。


NSTimer可能在主线程中创建,也可能在子线程中创建。主线程中的RunLoop默认是启动的,所以timer只要添加到主线程RunLoop中就会被执行;而子线程中的RunLoop默认是不启动的,所以timer添加到子线程RunLoop中后,还要手动启动RunLoop才能使timer被执行。


NSTimer只有添加到启动起来的RunLoop中才会正常运行。NSTimer通常不建议添加到主线程中执行,因为界面的更新在主线程中进行,这会影响NSTimer的准确性。

以下代码为4种情形下NSTimer的正确使用方法。


- (void)viewDidLoad {
    [super viewDidLoad];
    /*第一种,主线程中创建timer,需要手动添加到RunLoop中*/
    NSTimer *mainThreadTimer = [NSTimer timerWithTimeInterval:10.0 target:self selector:@selector(mainThreadTimer_SEL) userInfo:nil repeats:YES];
    /*第二种,主线程中创建timer,不需要手动添加到RunLoop中*/
    NSTimer *mainThreadSchduledTimer = [NSTimer scheduledTimerWithTimeInterval:10.0 target:self selector:@selector(mainThreadSchduledTimer_SEL) userInfo:nil repeats:YES];
    /*将mainThreadTimer添加到主线程runloop*/
    [[NSRunLoop currentRunLoop] addTimer:mainThreadTimer forMode:NSDefaultRunLoopMode];
    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
        /*第三种,子线程中创建timer,需要手动添加到RunLoop中*/
        NSTimer *subThreadTimer = [NSTimer timerWithTimeInterval:10.0 target:self selector:@selector(subThreadTimer_SEL) userInfo:nil repeats:YES];
        /*第四种,子线程中创建timer,不需要手动添加到RunLoop中*/
        NSTimer *subThreadScheduledTimer = [NSTimer scheduledTimerWithTimeInterval:10.0 target:self selector:@selector(subThreadSchduledTimer_SEL) userInfo:nil repeats:YES];
        /*将subThreadTimer添加到子线程runloop*/
        [[NSRunLoop currentRunLoop] addTimer:subThreadTimer forMode:NSDefaultRunLoopMode];
        /*启动子线程runloop*/
        [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
    });
- (void)maniThreadTimer_SEL{
    NSLog(@"mainThreadTimer is working!");
}
- (void)mainThreadScheduledTimer_SEL{
    NSLog(@"mainThreadScheduledTimer is working!");
}
- (void)subThreadTimer_SEL{
    NSLog(@"subThreadTimer is working!");
}
- (void)subThreadScheduledImer_SEL{
    NSLog(@"subThreadSchduledTimer is working!");
}


上述代码的打印结果为:


2466108-96a3f757aac4ce07.webp.jpg


NSTimer的释放方法:[timer invalidate];


5.CADisplayLink实现定时任务


CADisplayLink实现的定时器与屏幕刷新频率绑定在一起,是一种帧率刷新,适用于界面的不断重绘(例如流畅动画和视频播放等)。CADisplayLink以特定模式注册到RunLoop后,每当屏幕显示内容刷新结束后就会向CADisplayLink指定的target发送一次消息,实现target的每帧调用。根据需求也可以设置每几帧调用一次,默认每帧都调用。另外,通过CADisplayLink还可以获取帧率和时间等信息。


CADisplayLink实现的定时器精度非常高,但如果调用的方法十分耗时,超过一帧的时间间隔,那么会导致跳帧,跳帧次数取决于CPU的忙碌程度。


下面是CADisplayLink定时器的实现:


- (void)viewDidLoad {
    [super viewDidLoad];
    CADisplayLink *displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(displayLink_SEL)];
    /*添加到当前运行的RunLoop中启动*/
    [displayLink addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
    /*暂停、继续对selector的调用*/
//    [displayLink setPaused:YES];
//    [displayLink setPaused:NO];
    /*设置每几帧调用一次selector,默认为1*/
//    [displayLink setPreferredFramesPerSecond:2];
    /*移除,不再使用*/
//    [displayLink invalidate];
//    displayLink = nil;
}
- (void)displayLink_SEL{
    NSLog(@"displayLink is working!");
}


打印结果如下,每一帧都调用selector。


2466108-1c418eafb3649cce.webp.jpg


如何解决网络请求的依赖关系


当一个网络的请求需要依赖于另一个网络请求的结果时,如何解决网络请求的依赖关系?

假设一个网络请求B依赖于另一个网络请求A,即网络请求B需要网络请求A的结果,这就需要开发者采取措施来维护A和B的顺序,实现网络请求B同步等待网络请求A。思路主要有以下两个:


1)一种是通过线程管理来实现,即线程同步,让线程B等待线程A。iOS中实现线程同步的方法也很多:可以设置NSOperation操作依赖来实现,即让操作B依赖于操作A,操作B会在操作A结束后才会开始执行([B addDependency:A];);也可以通过GCD来实现,包括组队列(dispatch_group)、阻塞任务(dispatch_barrier_(a)sync)和信号量机制(dispatch_semaphore)。


2)另外也可以直接通过逻辑衔接来实现。即在网络请求A的响应回调中编写网络请求B的逻辑,缺点是耦合度上会高一些。


目录
相关文章
|
7月前
|
iOS开发
iOS多线程之NSOperationQueue-依赖、并发数、优先级、自定义Operation等最全的使用总结
iOS多线程之NSOperationQueue-依赖、并发数、优先级、自定义Operation等最全的使用总结
216 0
|
7月前
|
安全 调度 C语言
iOS多线程之GCD-同步、异步、并发、串行、线程组、栅栏函数、信号量等全网最全的总结
iOS多线程之GCD-同步、异步、并发、串行、线程组、栅栏函数、信号量等全网最全的总结
480 1
|
11月前
|
安全 算法 编译器
iOS线程安全——锁(二)
iOS线程安全——锁(二)
119 0
|
11月前
|
存储 安全 API
iOS线程安全——锁(一)
iOS线程安全——锁(一)
209 0
|
iOS开发
iOS多线程的初步研究-- dispatch同步
GCD提供两种方式支持dispatch队列同步,即dispatch组和信号量。
146 0
|
API iOS开发
iOS多个线程发起相同请求,避免重复
有时候在调用多个模块时,会对同一个API进行多次请求,但因为内容都是一样的,所以最好就是加上锁,防止重复请求造成网络资源浪费
152 0
|
敏捷开发 安全 Unix
iOS开发 - 在实战中挖掘之线程间的通信方式
iOS开发 - 在实战中挖掘之线程间的通信方式
153 0
|
存储 Java 数据库
JAVA实现网络多线程编程小游戏开发
实验总结:五子棋是一个很简单的游戏,但是如果认真对待,一个代码一个代码的去研究,会收获到很多知识,会打好学习基础。方便以后开发更高、更难的项目时打下稳固的基础。在自己开发的过程中会有各种意想不到的bug,通过查阅资料及询问老师同学进行解决对本身的一个代码能力会有一个质的增长,同时这也是一个非常快乐的过程。有进步,总归是好事。
JAVA实现网络多线程编程小游戏开发