iOS 多线程—GCD 基本用法

简介:

什么是进程?

最通俗的描述就是一个个pid,官方的说法:进程是程序在计算机上的一次执行活动。打开一个app 就开启了一个进程。可包含多个线程。

image

什么是线程?

独立执行的代码段,一个线程同一时间内只能执行一个任务,反之多线程并发就可以在同一时间执行多个任务。

同步和异步

一个同步函数只在完成了预定任务后才返回。会阻塞当前线程。异步时任务开启会立即返回,不阻塞当前线程去执行下一个函数。异步会开启其他线程。

串行和并发

串行:任务按先后顺序逐个执行。并发:后面的任务不会等前面的任务完成了再执行,同样会遵循先添加先执行的原则,但添加间隔往往忽略不计。所以看上去像是一起执行。

并发与并行

并发和并行通常被一起提到,所以值得花些时间解释它们之间的区别。

并发代码的不同部分可以“同步”执行。然而,该怎样发生或是否发生都取决于系统。多核设备通过并行来同时执行多个线程;然而,为了使单核设备也能实现这一点,它们必须先运行一个线程,执行一个上下文切换,然后运行另一个线程或进程。这通常发生地足够快以致给我们并发执行地错觉,如下图所示:

image

虽然你可以编写代码在 GCD 下并发执行,但 GCD 会决定有多少并行的需求。并行要求并发,但并发并不能保证并行。

什么是GCD?

GCD 是一套低层API,用于将任务切分成单一任务提交至队列并发或者串行执行。遵循FIFO 原则,先提交到队列的先执行。串行队列和并发队列都是如此。

串行队列

串行队列中的任务一次执行一个,每个任务只在前一个任务完成时才开始。而且,你不知道在一个 Block 结束和下一个开始之间的时间长度,如下图所示: image

并发队列

在并发队列中的任务能得到的保证是它们会按照被添加的顺序开始执行,但这就是全部的保证了。任务可能以任意顺序完成,你不会知道何时开始运行下一个任务,或者任意时刻有多少 Block 在运行。再说一遍,这完全取决于 GCD 。

下图展示了一个示例任务执行计划,GCD 管理着四个并发任务:

image

GCD基本队列类型

  1. Main quene

主线程队列,串行,可以通过dispatch_get_main_quene() 获取。UI操作都需要在主线程中执行。

  1. Global quene

系统提供的并发队列。通过dispatch_get_global_queue 创建。

  1. Custom quene

自定义队列,可以为串行,也可为并发。通过dispatch_queue_create 创建。

队列组

将多线程进行分组,最大的好处是可获知所有线程的完成情况。当多线程并发执行时,由于单个线程什么时候结束并不知道,所以很难判断线程组整个完成情况,通过dispatch_group_notify,可以直接监听组里所有线程完成情况。

常规用法

  1. Global quene 及 Custom quene(创建串行队列)

1.1 并发队列,异步执行

此处为直接使用global_quene

    override func viewDidLoad() {
        super.viewDidLoad()
        // 并发队列,异步执行
        for index in 1...5 {
            dispatch_async(dispatch_get_global_queue(0, 0), { () -> Void in
                //                println("currentIndex----\(index)")
                NSLog("currentIndex:----\(index), thread:---%@",NSThread.currentThread())
            })
        }
        NSLog("主线程,thread:---%@",NSThread.currentThread())
    }
AI 代码解读

NSLog打印结果为:

image

可看到执行完成为无序的,而且每次都不一样。同样也能看到出主线程外,另外开启了5个线程。 注意此处用的是NSLog 输出,而不是Println。因为NSLog 本身是同步的,而Println 为异步,在多线程并发调用时Println 输出结果会错乱。

1.2 并发队列,同步执行

还是上面的例子,紧改为同步执行dispatch_sync

    override func viewDidLoad() {
        super.viewDidLoad()
        // 并发队列,同步执行
        for index in 1...5 {
            dispatch_sync(dispatch_get_global_queue(0, 0), { () -> Void in
                //                println("currentIndex----\(index)")
                NSLog("currentIndex:----\(index), thread:---%@",NSThread.currentThread())
            })
        }
        NSLog("主线程,thread:---%@",NSThread.currentThread())
    }
AI 代码解读

打印结果为:

image

可看到并没有开启其他线程,任务按顺序逐个执行,同时阻塞主线程。搞不懂这种“并发队列,同步执行”的意义所在。

1.3 串行队列,异步执行

使用dispatch_quene_creat 创建串行队列

    override func viewDidLoad() {
        super.viewDidLoad()
        // 串行队列,异步执行
        var quene  = dispatch_queue_create("1", DISPATCH_QUEUE_SERIAL)// 创建串行队列
        for index in 1...5 {
            dispatch_async(quene, { () -> Void in
                //                println("currentIndex----\(index)")
                NSLog("currentIndex:----\(index), thread:---%@",NSThread.currentThread())
            })
        }
        NSLog("主线程,thread:---%@",NSThread.currentThread())
    }
AI 代码解读

打印结果为:

image

可以看到另外开启了一个线程,不会将主线程阻塞,任务按顺序执行。

1.4 串行队列,同步执行

使用dispatch_quene_creat 创建串行队列

    override func viewDidLoad() {
        super.viewDidLoad()
        // 串行队列,同步执行
        for index in 1...5 {
            dispatch_sync(quene, { () -> Void in
                //                println("currentIndex----\(index)")
                NSLog("currentIndex:----\(index), thread:---%@",NSThread.currentThread())
            })
        }
        NSLog("主线程,thread:---%@",NSThread.currentThread())
    }
AI 代码解读

打印结果为: image可看到不会开启其他线程,会阻塞主线程,任务按顺序执行。

1.5 Custom quene 创建并发队列

还是一样的例子,只不过改为

        var quene  = dispatch_queue_create("1",DISPATCH_QUEUE_CONCURRENT)// 创建并发队列
AI 代码解读

可看到异步,同步执行结果与1.1 和1.2 一样。就不一一列出了。

  1. Main quene 使用,线程死锁

想必这个应该都知道怎么用,在其他线程中回到主线程,去执行ui操作。注意是在其他线程中获取主线程。所以要注意以下问题。

2.1 不要在主线程中获取主线程队列,并同步执行任务。

    override func viewDidLoad() {
        dispatch_sync(dispatch_get_main_queue(), { () -> Void in
            NSLog("在主线程执行任务")
        })
    }
AI 代码解读

这种写法一定会线程死锁。同步执行首先就阻塞了主线程,然后又想在主线程去执行任务所以任务没法完成,任务没法完成又导致了线程没法结束。所以导致了恶性循环,主线程就一直这么阻塞着。导致UI一直卡住。

  1. 队列组

3.1 使用场景

个人觉得先要知道什么时候需要使用到队列组。队列组一般配合dispatch_group_notify 使用,用于监听这一组任务是否全部完成。所以使用场景为:

你要有多个任务,如果是单个任务的情况,根本没有必要使用队列组。
而且还要是异步执行的情况,若是同步阻塞在那执行完了自然知道。
AI 代码解读

同时也不要认为队列组就会有很多队列,其实不是,队列组其实是要实现的是对线程所有任务的分组监听,所以只有一个队列也可以

3.1.1异步执行,串行队列组

    override func viewDidLoad() {
        super.viewDidLoad()
        var group = dispatch_group_create()
        var quene = dispatch_queue_create("1", DISPATCH_QUEUE_SERIAL)//串行队列

        dispatch_group_notify(group, quene) { () -> Void in
            dispatch_async(dispatch_get_main_queue(), { () -> Void in
                NSLog("任务结束,回到主线程")
            })
        }
        for index in 1...5 {
            dispatch_group_async(group, quene, { () -> Void in
                NSLog("currentIndex:----\(index), thread:---%@",NSThread.currentThread())
            })
        }
        NSLog("主线程")
    }
AI 代码解读

打印结果为:

image

可看到除主线程外,有其他一个线程,串行任务结束后能够实时监听到,回到主线程。

3.1.2异步执行,并发队列组

    override func viewDidLoad() {
        super.viewDidLoad()
        var group = dispatch_group_create()
        var quene = dispatch_queue_create("1", DISPATCH_QUEUE_CONCURRENT)//并发队列,也可以用global_quene
        dispatch_group_notify(group, quene) { () -> Void in
            dispatch_async(dispatch_get_main_queue(), { () -> Void in
                NSLog("任务结束,回到主线程")
            })
        }
        for index in 1...5 {
            dispatch_group_async(group, quene, { () -> Void in
                NSLog("currentIndex:----\(index), thread:---%@",NSThread.currentThread())
            })
        }
        NSLog("主线程")
    }
AI 代码解读

打印结果为:

image

同样所有线程的任务全部结束后,能监听到,通知主线程。

写在最后

本文凭借个人理解,同时也参考了不少博客。列举了GCD的一些常规用法,还有很多特性没有一一列出。不熟的知识点不敢乱写,以后再慢慢完善。iOS 多线程也不仅仅是GCD 这一种,但我觉得最好用的还是GCD。觉得有用就打赏,点赞,同时有错误忘指正...

文章转载自 开源中国社区[https://www.oschina.net]

目录
打赏
0
0
0
0
16427
分享
相关文章
安卓与iOS开发中的线程管理差异解析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自拥有独特的魅力。如同东西方文化的差异,它们在处理多线程任务时也展现出不同的哲学。本文将带你穿梭于这两个平台之间,比较它们在线程管理上的核心理念、实现方式及性能考量,助你成为跨平台的编程高手。
深入探索Android与iOS的多线程编程差异
在移动应用开发领域,多线程编程是提高应用性能和响应性的关键。本文将对比分析Android和iOS两大平台在多线程处理上的不同实现机制,探讨它们各自的优势与局限性,并通过实例展示如何在这两个平台上进行有效的多线程编程。通过深入了解这些差异,开发者可以更好地选择适合自己项目需求的技术和策略,从而优化应用的性能和用户体验。
安卓与iOS开发中的线程管理对比
【9月更文挑战第12天】在移动应用的世界中,安卓和iOS平台各自拥有庞大的用户群体。开发者们在这两个平台上构建应用时,线程管理是他们必须面对的关键挑战之一。本文将深入探讨两大平台在线程管理方面的异同,通过直观的代码示例,揭示它们各自的设计理念和实现方式,帮助读者更好地理解如何在安卓与iOS开发中高效地处理多线程任务。
iOS开发-多线程编程
【8月更文挑战第12天】在iOS开发中,属性的内存管理至关重要,直接影响应用性能与稳定性。主要策略包括:`strong`(强引用),保持对象不被释放;`weak`(弱引用),不保持对象,有助于避免循环引用;`assign`(赋值),适用于基本数据类型及非指针对象类型;`copy`(复制),复制对象而非引用,确保不变性。内存管理基于引用计数,利用自动引用计数(ARC)自动管理对象生命周期。此外,需注意避免循环引用,特别是在block中。最佳实践包括理解各策略、避免不必要的强引用、及时释放不再使用的对象、注意block中的内存管理,并使用工具进行内存分析。正确管理内存能显著提升应用质量。
|
6月前
|
探索iOS开发:Swift中的异步编程与GCD应用
【8月更文挑战第4天】在iOS开发的海洋中,掌握Swift语言的航向是至关重要的。本文将引领你深入理解Swift中的异步编程概念,并借助Grand Central Dispatch(GCD)这一强大的工具,来简化并发编程的复杂性。我们将通过实际代码示例,展现如何在iOS应用中高效地管理后台任务和提升用户界面的响应性。
114 3
网络安全与信息安全:关于网络安全漏洞、加密技术、安全意识等方面的知识分享安卓与iOS开发中的线程管理比较
【8月更文挑战第30天】本文将探讨网络安全与信息安全的重要性,并分享关于网络安全漏洞、加密技术和安全意识的知识。我们将了解常见的网络攻击类型和防御策略,以及如何通过加密技术和提高安全意识来保护个人和组织的信息安全。
iOS 16 系统键盘修复问题之确定_lock是否用于保护对_deferredTasks的多线程读写如何解决
iOS 16 系统键盘修复问题之确定_lock是否用于保护对_deferredTasks的多线程读写如何解决
Java从入门到精通:3.2.1分布式与并发编程——深入Java并发包,精通多线程高级用法
Java从入门到精通:3.2.1分布式与并发编程——深入Java并发包,精通多线程高级用法
109 1
|
9月前
|
Java多线程基础-7:wait() 和 notify() 用法解析
这篇内容探讨了Java中的`wait()`和`notify()`方法在多线程编程中的使用。
63 0

热门文章

最新文章

  • 1
    Cellebrite UFED 4PC 7.71 (Windows) - Android 和 iOS 移动设备取证软件
    24
  • 2
    【03】仿站技术之python技术,看完学会再也不用去购买收费工具了-修改整体页面做好安卓下载发给客户-并且开始提交网站公安备案-作为APP下载落地页文娱产品一定要备案-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
    33
  • 3
    【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
    28
  • 4
    【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
    23
  • 5
    uniapp开发ios打包Error code = -5000 Error message: Error: certificate file(p12) import failed!报错问题如何解决
    143
  • 6
    【05】2025年1月首发完整版-篇幅较长-苹果app如何上架到app store完整流程·不借助第三方上架工具的情况下无需花钱但需仔细学习-优雅草央千澈详解关于APP签名以及分发-们最关心的一篇来了-IOS上架app
    233
  • 7
    app开发之安卓Android+苹果ios打包所有权限对应解释列表【长期更新】-以及默认打包自动添加权限列表和简化后的基本打包权限列表以uniapp为例-优雅草央千澈
    90
  • 8
    深入探索iOS开发中的SwiftUI框架
    143
  • 9
    ios样式开关按钮jQuery插件
    58
  • 10
    Android与iOS生态差异深度剖析:技术架构、开发体验与市场影响####
    75