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的逻辑,缺点是耦合度上会高一些。


目录
相关文章
|
10月前
|
Java API 微服务
为什么虚拟线程将改变Java并发编程?
为什么虚拟线程将改变Java并发编程?
427 83
|
7月前
|
Java
如何在Java中进行多线程编程
Java多线程编程常用方式包括:继承Thread类、实现Runnable接口、Callable接口(可返回结果)及使用线程池。推荐线程池以提升性能,避免频繁创建线程。结合同步与通信机制,可有效管理并发任务。
282 6
|
10月前
|
负载均衡 算法 安全
基于Reactor模式的高性能网络库之线程池组件设计篇
EventLoopThreadPool 是 Reactor 模式中实现“一个主线程 + 多个工作线程”的关键组件,用于高效管理多个 EventLoop 并在多核 CPU 上分担高并发 I/O 压力。通过封装 Thread 类和 EventLoopThread,实现线程创建、管理和事件循环的调度,形成线程池结构。每个 EventLoopThread 管理一个子线程与对应的 EventLoop(subloop),主线程(base loop)通过负载均衡算法将任务派发至各 subloop,从而提升系统性能与并发处理能力。
495 3
|
7月前
|
安全 Linux 虚拟化
Cisco IOS XRv 9000 Router IOS XR Release 7.11.2 MD - 思科 IOS XR 网络操作系统
Cisco IOS XRv 9000 Router IOS XR Release 7.11.2 MD - 思科 IOS XR 网络操作系统
403 3
Cisco IOS XRv 9000 Router IOS XR Release 7.11.2 MD - 思科 IOS XR 网络操作系统
|
12月前
|
机器学习/深度学习 消息中间件 存储
【高薪程序员必看】万字长文拆解Java并发编程!(9-2):并发工具-线程池
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的强力并发工具-线程池,废话不多说让我们直接开始。
415 0
|
7月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
347 1
|
7月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
343 1
|
8月前
|
算法 Java
Java多线程编程:实现线程间数据共享机制
以上就是Java中几种主要处理多线程序列化资源以及协调各自独立运行但需相互配合以完成任务threads 的技术手段与策略。正确应用上述技术将大大增强你程序稳定性与效率同时也降低bug出现率因此深刻理解每项技术背后理论至关重要.
507 16
|
7月前
|
Java 调度 数据库
Python threading模块:多线程编程的实战指南
本文深入讲解Python多线程编程,涵盖threading模块的核心用法:线程创建、生命周期、同步机制(锁、信号量、条件变量)、线程通信(队列)、守护线程与线程池应用。结合实战案例,如多线程下载器,帮助开发者提升程序并发性能,适用于I/O密集型任务处理。
685 0
|
存储 网络协议 安全
Java网络编程,多线程,IO流综合小项目一一ChatBoxes
**项目介绍**:本项目实现了一个基于TCP协议的C/S架构控制台聊天室,支持局域网内多客户端同时聊天。用户需注册并登录,用户名唯一,密码格式为字母开头加纯数字。登录后可实时聊天,服务端负责验证用户信息并转发消息。 **项目亮点**: - **C/S架构**:客户端与服务端通过TCP连接通信。 - **多线程**:采用多线程处理多个客户端的并发请求,确保实时交互。 - **IO流**:使用BufferedReader和BufferedWriter进行数据传输,确保高效稳定的通信。 - **线程安全**:通过同步代码块和锁机制保证共享数据的安全性。
596 23

热门文章

最新文章