RunLoop 详解

概述:

RunLoop 是一个核心的 iOS 机制,它是 事件循环机制 的实现,负责管理线程的执行和调度。RunLoop 能够持续监听输入事件(如用户触摸、定时器、网络请求等)并分发给相应的处理方法,从而保持应用程序持续运行。

核心功能:

  1. 处理异步事件: RunLoop 可以管理事件源,比如定时器、触摸事件、网络响应等,确保这些事件在合适的时机被触发。
  2. 控制线程执行: 通过控制线程是否处于休眠或运行状态,RunLoop 能够让线程处于活跃状态,并等待事件的发生。
  3. 保持线程活跃: 如果没有事件需要处理,RunLoop 会让线程进入休眠状态,节省系统资源。

1. 为什么需要 RunLoop?

背景:

在多线程编程中,主线程(UI线程)必须保持活跃,才能响应用户的操作和刷新界面。没有 RunLoop,即便你创建了一个线程,也不能有效地等待和处理外部事件。

  • UI更新:当我们触发按钮点击等操作时,事件会通过 RunLoop 传递给相应的处理方法,确保 UI 在需要的时候得到更新。
  • 后台任务:在后台线程处理任务时,我们也需要 RunLoop 来监听定时器、网络请求等异步任务。

没有 RunLoop,线程就像失去了生命,即使任务没完成,它也会直接退出。

2. RunLoop 的基本工作原理

基本流程:

  1. RunLoop 启动:线程启动后,会进入 RunLoop 循环。
  2. 等待事件RunLoop 会等待外部事件的发生,比如触摸事件、定时器触发、网络回调等。
  3. 事件处理:当事件发生时,RunLoop 会将事件分发到对应的处理方法(例如响应按钮点击事件、定时器回调等)。
  4. 继续运行:当事件处理完后,RunLoop 会继续等待新的事件,或者退出。

RunLoop 状态:

  • 休眠状态:当没有事件需要处理时,RunLoop 处于休眠状态。休眠时,线程不会消耗过多的 CPU 资源。
  • 活跃状态:当有事件发生时,RunLoop 被唤醒,进入活跃状态来处理这些事件。

3. 常见用法

(1) 主线程的 RunLoop

主线程的 RunLoop 是自动开启的,常用于UI事件处理异步任务的等待

例如,在处理定时器时,你可能希望在主线程中定期执行某些操作:

NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(updateUI) userInfo:nil repeats:YES];
[[NSRunLoop mainRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
  • 定时器会定期触发 updateUI 方法。
  • NSRunLoopCommonModes 确保定时器在滚动、拖动等事件中也能继续触发。

(2) 自定义线程的 RunLoop

对于自定义线程(非主线程),你需要手动创建并启动 RunLoop

- (void)startCustomThread {NSThread *customThread = [[NSThread alloc] initWithTarget:self selector:@selector(runThread) object:nil];[customThread start];
}- (void)runThread {@autoreleasepool {// 创建并启动 RunLoop[[NSRunLoop currentRunLoop] run]; // 必须显式调用,线程才能持续运行}
}

[[NSRunLoop currentRunLoop] run] 启动线程的 RunLoop,使其持续运行,处理事件。

(3) 使用 RunLoop 处理定时器

NSTimer *timer = [NSTimer timerWithTimeInterval:2.0 target:self selector:@selector(handleTimer) userInfo:nil repeats:YES];
[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSRunLoopCommonModes];
  • 这样可以确保定时器的回调不会被滚动等事件阻塞。
  • NSRunLoopCommonModes 模式确保在用户滚动屏幕时,定时器依然可以触发。

4. 解决实际问题:

(1) 主线程阻塞问题

RunLoop 可以有效避免主线程阻塞。很多时候我们需要执行耗时操作(比如网络请求),但不能让主线程被阻塞,否则界面无法响应。

- (void)fetchData {NSURL *url = [NSURL URLWithString:@"https://example.com"];NSURLRequest *request = [NSURLRequest requestWithURL:url];NSURLSessionDataTask *task = [[NSURLSession sharedSession] dataTaskWithRequest:request completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {// 网络请求完成后,更新 UIdispatch_async(dispatch_get_main_queue(), ^{// 更新 UI});}];[task resume];
}
  • 通过 dispatch_async 把 UI 更新放到主线程,避免主线程阻塞。
  • 事件循环 让主线程保持活跃,直到请求完成,才处理回调。

(2) 后台任务与定时器的配合

当我们在后台线程执行定时任务时,必须确保线程运行不被中断。使用 RunLoop 能保证后台线程的活跃:

- (void)performBackgroundTask {[NSThread detachNewThreadSelector:@selector(executeBackgroundTask) toTarget:self withObject:nil];
}- (void)executeBackgroundTask {@autoreleasepool {NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(backgroundTask) userInfo:nil repeats:YES];[[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode];[[NSRunLoop currentRunLoop] run];  // 保持线程活跃}
}- (void)backgroundTask {NSLog(@"Background task executed");
}

通过 RunLoop 保证定时器在后台线程中正常触发,否则后台线程执行完毕后会自动退出,定时器就会停止。

5. 注意事项与优缺点

优点:

  • 节省系统资源:通过在没有事件时让线程进入休眠状态,RunLoop 可以有效减少不必要的 CPU 消耗。
  • 异步任务管理:能够很方便地在多线程中管理事件源(如定时器、网络请求等),确保线程在等待事件时不会结束。
  • 灵活的线程控制RunLoop 提供了灵活的方式来控制线程,确保线程在处理事件时能一直活跃。

缺点:

  • 不能主动退出RunLoop 只会在有事件时才会退出,如果没有事件源,线程会一直保持休眠状态,因此我们需要手动管理退出条件。
  • 性能问题:过多的事件源可能会增加线程的调度开销,影响应用的性能。
  • 内存管理RunLoop 本身并不负责对象的内存管理,所以在使用时要注意内存泄漏(例如通过定时器等引用)。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/17936.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2. grafana插件安装并接入zabbix

一、在线安装 如果不指定安装位置,则默认安装位置为/var/lib/grafana/plugins 插件安装完成之后需要重启grafana 命令在上一篇讲到过 //查看相关帮助 [rootlocalhost ~]# grafana-cli plugins --help //从列举中的插件过滤zabbix插件 [rootlocalhost ~]# grafana…

【Linux】Ubuntu Linux 系统——Node.js 开发环境

ℹ️大家好,我是练小杰,今天星期五了,同时也是2025年的情人节,今晚又是一个人的举个爪子!! 🙂 本文是有关Linux 操作系统中 Node.js 开发环境基础知识,后续我将添加更多相关知识噢&a…

DeepSeek全方位解读:模型介绍,优势及应用场景

DeepSeek全方位解读:领先科技背后的革新力量 前言1.DeepSeek整体介绍2.DeepSeek-R13.DeepSeek-V34.DeepSeek系列模型之间的关系5.Deepseek优势及应用场景6.模型参数与量化精度的关系7.行业部署Deepseek及应用情况 前言 在当今快速发展的科技世界里,人工…

电脑端调用摄像头拍照:从基础到实现

文章目录 1. 了解navigator.mediaDevices.getUserMedia API2. 创建 HTML 结构3. 编写 JavaScript 代码3.1 打开摄像头3.2 拍照 4. 完整代码5. 测试6. 注意事项及部署 在现代 Web 开发中,调用摄像头进行拍照是一个常见的功能,尤其是在需要用户上传头像、进…

windows平台上 oracle简单操作手册

一 环境描述 Oracle 11g单机环境 二 基本操作 2.1 数据库的启动与停止 启动: C:\Users\Administrator>sqlplus / as sysdba SQL*Plus: Release 11.2.0.4.0 Production on 星期五 7月 31 12:19:51 2020 Copyright (c) 1982, 2013, Oracle. All rights reserved. 连接到:…

Java面试——Tomcat

优质博文:IT_BLOG_CN 一、Tomcat 顶层架构 Tomcat中最顶层的容器是Server,代表着整个服务器,从上图中可以看出,一个Server可以包含至少一个Service,用于具体提供服务。Service主要包含两个部分:Connector和…

3. 导入官方dashboard

官方dashboard:https://grafana.com/grafana/dashboards 1. 点击仪表板 - 新建 - 导入 注:有网络的情况想可以使用ID,无网络情况下使用仪表板josn文件 2. 在官方dashboard网页上选择符合你现在数据源的dashboard - 点击进入 3. 下拉网页选…

你如何利用SIMD(如SSE/AVX)优化图像处理的性能?

SIMD优化问题 1. SIMD 在图像处理中的优化方式2. 典型应用场景3. SIMD 的常见优化技巧4. 总结 利用 SIMD(Single Instruction, Multiple Data) 指令集(如 SSE/AVX/AVX2/AVX-512)优化图像处理的性能,可以极大地提升计算…

高并发场景下,如何用无锁实现高性能LRU缓存?

《百万人高并发场景下,我如何用无锁实现高性能LRU缓存?》 LRU算法核心原理 LRU(Least Recently Used)算法是缓存系统的核心淘汰策略,其核心逻辑可以用一张流程图描述: (图:访问数…

HAL库框架学习总结

概述:HAL库为各种外设基本都配了三套 API,查询,中断和 DMA。 一、HAL库为外设初始化提供了一套框架,这里以串口为例进行说明,调用函数 HAL_UART_Init初始化串口,此函数就会调用 HAL_UART_MspInit&#xff0…

LAWS是典型的人机环境系统

致命性自主武器系统(Lethal Autonomous Weapons Systems,LAWS)是一种典型的人机环境系统,它通过高度集成的传感器、算法和武器平台,在复杂的战场环境中自主执行任务。LAWS能够自主感知环境、识别目标、做出决策并实施攻…

【16届蓝桥杯寒假刷题营】第1期DAY4

4.可达岛屿的个数 - 蓝桥云课 题目背景 在一个神奇的魔法世界中,有一座古老的迷幻之城。迷幻之城被分成 n 个鸟屿,编号从 1 到 n,共有 m 座桥。迷幻之城的居民们希望能够建立起紧密的联系,每个岛屿上的居民都想知道自己最多能到…

【物联网】电子电路基础知识

文章目录 一、基本元器件1. 电阻2. 电容3. 电感4. 二极管(1)符号(2)特性(3)实例分析5. 三极管(1)符号(2)开关特性(3)实例6. MOS管(产效应管)(1)符号(2)MOS管极性判定(3)MOS管作为开关(4)MOS管vs三极管7. 门电路(1)与门(2)或门(3)非门二、常用元器件…

数据结构 04

4. 栈 4.2. 链式栈 4.2.1. 特性 逻辑结构:线性结构 存储结构:链式存储结构 操作:创建,入栈,出栈,清空,获取 4.2.2. 代码实现 头文件 LinkStack.h #ifndef __LINKSTACK_H__ #define __LINKST…

【云安全】云原生-K8S(四)安全问题分析

Kubernetes(K8S)因其强大的容器编排能力成为了云计算和微服务架构的首选,但同时也带来了复杂的安全挑战。本文将概述K8S的主要安全问题,帮助安全工程师理解潜在威胁,并采取相应的防护措施。 K8S 攻击面概览 下面两张…

【Unity新手】Text不显示字的问题解决办法

很多同学在unity里导入了一个Text发现字没有显示出来为什么呢? 首先在网络上下载一个.ttf或者.otf字体文件,导入资源,比如说我下载了黑体.otf 然后导入unity,右键字体TextMesgPro-FontAsset 然后字体设置里添加上就可以了

基于Flask的影视剧热度数据可视化分析系统的设计与实现

【FLask】基于Flask的影视剧热度数据可视化分析系统的设计与实现(完整系统源码开发笔记详细部署教程)✅ 目录 一、项目简介二、项目界面展示三、项目视频展示 一、项目简介 随着互联网技术的飞速发展,影视剧行业的数据量呈爆炸性增长&#x…

React 低代码项目:组件设计

React 低代码项目:组件设计 Date: February 6, 2025 React表单组件 **目标:**使用 Ant Design 表单组件,开发登录、注册、搜索功能 内容: 使用 React 表单组件、受控组件使用 Ant Design 表单组件使用 表单组件的校验和错误提…

vue-plugin-hiprint (vue2

页面效果 <template><div><div class="d-flex flex-column mt5"><div class="d-flex flex-row " style="margin-bottom: 10px;justify-content: center;"><!-- 纸张大小 A3、A4 等 --><div class="paper…

C++17 中的 std::reduce:详细教程

文章目录 1. 简介2. 函数签名3. 使用场景3.1 简单的累加操作3.2 自定义归并操作3.3 并行计算的性能优势 4. 注意事项4.1 归并操作的结合律和交换律4.2 默认值的使用 5. 总结 1. 简介 std::reduce 是 C17 标准库中引入的一个算法&#xff0c;用于对范围内的元素进行归并操作。它…