【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记31 Multithreading多线程

在IOS中存在着许多队列,和我们数据结构中的队列一样,这里的队列概念也是先进先出的。而每一个方法(包括闭包)都被组织在这些不同的队列中,而每一个队列都有自己的线程去运行这些队列,这就造就了多线程环境。

其中有一个非常重要的队列叫做主队列,主队列是一个串行队列,所以主队列只会一个一个地执行主队列中的函数。所有的UI活动都必须发生在主队列中,所以当你想要一个函数或者是闭包的时候就会执行某些代码,这就会做任何关于UI的事,必须把它放到主队列中,这是保护UI的好办法,主队列绝对不想做任何可能被阻塞的事情,比如读取一个包含URL的NSData,上一话中的Demo当我们点击按钮获取大图的时候会卡顿,所以我们要把它从主队列中拿出来。

通常你使用MVC中的东西都在主队列中,比如页面生命周期中的ViewDidLoad、viewWillAppear,我们不需要特别的留心。

IOS会在你需要的时候为你创建其他队列。

那么如何在其他队列中执行函数或者闭包呢?

首先初始化一个队列,然后调用了函数dispatch_async,有两个参数,第一个是队列,第二个是一个闭包,这里写成了尾随闭包的形式。那么第一句中如何获得主队列呢?使用函数dispatch_get_main_queue(),就能返回主队列了。此外还有一个面向对象的方法可以做到:NSOperationQueue.mainQueue()。上面有一个例子,在其他队列中执行一些可能阻塞UI的操作,然后打开主队列,执行需要呈现给UI的东西。

下面介绍一些主队列之外的其他队列,这些队列通常使用不同的处理级别来表示,处理级别的高低也代表了队列不同的处理速度。

要创建一个非主队列,首先要创建一个服务级别,创建级别的方法有点奇怪,使用了一个Int的构造器,构造其中选择级别的value属性,这种做法是由历史原因造成的。在创建好级别之后,使用dispatch_get_global_queue函数,传入级别和一个0,这个0以后会用到。有了新的队列,你就可以使用dispatch_asnyc函数并且在非主队列中执行代码了。

如果有多核处理器,队列真的可以并行运行,但是大部分情况下他们是分时的。你可以创建自己的串行队列,使用dispatch_queue_create来建立一个串行的队列。

在iOS中有些API是多线程的,比如下面这个例子:

比如上例中的NSURLSession这个类,注意黄色字体的部分,它让你从一个URL中下载一个文件,而且是异步的去做。在方法后面增加了一个闭包,这个闭包的作用是当你下载完毕时你需要在UI中打开这个文件,闭包为你指定了一个本地URL,一个HTTP应答了一个错误信息,利用这些信息你会做一些更新UI的事情。那么这样做行得通么?答案是“NO”。

因为这个下载的文件不在主队列中,解决办法是在闭包中打开主队列:

现在来进行实战环节,回到我们之前的Demo中,打开图片总是会造成阻塞,这并不好,尝试使用多线程的知识来解决这个问题。

我们需要做的是把获取图片的环节放到其他线程中,方法fetchImage修改如下:

 func fetchImage(){
        if let url = imageURL {
        let qos = Int(QOS_CLASS_USER_INITIATED.value)//指定服务“可能花费时间,但是用户需要得到,尽快完成”
        dispatch_async(dispatch_get_global_queue(qos, 0)){ () ->Void in//把要执行的代码放到闭包中
        let imageData = NSData(contentsOfURL: url)
        if imageData != nil{
        self.image = UIImage(data: imageData!)
        } else {
        self.image = nil
        }
            }
        }
    }

注意现在还不够,设置image的动作会修改UI,所以这个动作应该在主队列中,我们把设置image的动作放到主队列中:

 func fetchImage(){
        if let url = imageURL {
        let qos = Int(QOS_CLASS_USER_INITIATED.value)//指定服务“可能花费时间,但是用户需要得到,尽快完成”
        dispatch_async(dispatch_get_global_queue(qos, 0)){ () ->Void in//把要执行的代码放到闭包中
        let imageData = NSData(contentsOfURL: url)//加载图片应该放到其他线程中
            dispatch_async(dispatch_get_main_queue()){//把加载到的图片显示在页面中的时候使用主线程
        if imageData != nil{
        self.image = UIImage(data: imageData!)
        } else {
        self.image = nil
        }
            }
            }
        }
    }

现在加载图片的时候系统不会卡顿,你会跳转到下一个页面中,在当前页面中等待即可看到加载完的图片:

现在的问题是每一次点击一个按钮都会生成一个全新的MVC,那么如何避免重复加载呢,我们需要判断一下点击按钮时获取的URL是否是当前页面上图片的URL,方法修改如下:

  func fetchImage(){
        if let url = imageURL {
        let qos = Int(QOS_CLASS_USER_INITIATED.value)//指定服务“可能花费时间,但是用户需要得到,尽快完成”
        dispatch_async(dispatch_get_global_queue(qos, 0)){ () ->Void in//把要执行的代码放到闭包中
        let imageData = NSData(contentsOfURL: url)//加载图片应该放到其他线程中
            dispatch_async(dispatch_get_main_queue()){//把加载到的图片显示在页面中的时候使用主线程
                if url == self.imageURL{//新增判断语句
        if imageData != nil{
        self.image = UIImage(data: imageData!)
        } else {
        self.image = nil
        }
            }
            }
            }
        }
    }

最后需要做的是在其他队列加载图片的时候,主页面上运行一个齿轮来表示这个加载过程。

现在去对象库中拖出一个齿轮控件到场景中,打开文档大纲你会发现这个新增的齿轮控件被加到了scrollview中,这是因为我们之前用这个scrollview几乎铺满了我们的view:

这里文档大纲的优势就体现出来了,你可以在文档大纲中拖动控件来组合它们的次序,下面是正确地顺序:

另外在文档大纲中可以增加约束,和在IB中一样,按住control拖动即可,我们让齿轮控件居中

然后设置齿轮控件的属性,勾选当它停止时消失的选项:

和控制器建立连接:

@IBOutlet weak var spinner: UIActivityIndicatorView!

当我建立一个HTTP请求的时候让这个齿轮转动,当我结束这个请求的时候,让齿轮停止转动。

所以很明显我们需要在下面的位置增加齿轮的转动:

 func fetchImage(){
        if let url = imageURL {
            spinner?.startAnimating()//新增齿轮转动
        let qos = Int(QOS_CLASS_USER_INITIATED.value)//指定服务“可能花费时间,但是用户需要得到,尽快完成”
        dispatch_async(dispatch_get_global_queue(qos, 0)){ () ->Void in//把要执行的代码放到闭包中
        let imageData = NSData(contentsOfURL: url)//加载图片应该放到其他线程中
            dispatch_async(dispatch_get_main_queue()){//把加载到的图片显示在页面中的时候使用主线程
                if url == self.imageURL{//新增判断语句
        if imageData != nil{
        self.image = UIImage(data: imageData!)
        } else {
        self.image = nil
        }
            }
            }
            }
        }
    }

注意在调用spinner的时候把它当做可选型,因为页面可能先于spinner控件生成,在调用控件的时候这是个常用的办法。但是这个方法中加载完毕有不同的处理情况,共同点是都要处理image,所以我们选择在计算属性image的set方法中停止齿轮:

private var image:UIImage? {
        get {return imageView.image}
        set {
        imageView.image = newValue
        imageView.sizeToFit()
        scrollview?.contentSize = imageView.frame.size
        spinner?.stopAnimating()//加载完成了停止齿轮
        }
    }

现在来运行时候,当图片没有加载完成的时候,之前我们看到的是一个白色的页面,现在页面中间有齿轮在转动了:

加载完毕后齿轮消失:

时间: 2024-08-11 17:28:30

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记31 Multithreading多线程的相关文章

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记1 IOS8概述

首先感谢网易公开课和SwiftV课堂的朋友们辛苦翻译,这个系列是我学习斯坦福IOS8公开课的个人心得体会和笔记,希望能给大家带来启发. 首先我们要知道IOS系统中的结构情况,从贴近硬件的底层到贴近用户的顶层,分为四个层次: 1.Core OS层在最下层,很多人可能不知道IOS是一个基于UNIX的操作系统,它大量借鉴了Mac os X 的内核部分,Mac OS X我们肯定不会陌生,命令行的使用很好的证明了它是一个基于UNIX的系统.IOS针对移动设备对电池等硬件进行了系统的优化,但它仍可被看成是一

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记2 Xcode、Auto Layout及MVC

原文链接不知道在哪, 接着上一话来讲,上一话中讲到了MVC,那么MVC在IOS8开发中是如何应用的呢?Paul Hegarty老师给我们展示了一个计算器的Demo,首先新建一个工程,老师把AppDelegate.swift.LaunchScreen.xib和Images.xcassests文件放到了supporting Files文件夹中,那么剩下的两个文件ViewController.swift就是MVC中的C(控制器),Main.storyboard就是MVC中的V(视图). 在Main.s

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记13 Drawing绘制、UIColor颜色、Fonts字体

上一话介绍了视图绘制的一些基本原理,这一话继续展开.UIBezierPath可以绘制许多有趣的图形. 使用不同的构造器,比如roundedRect就是四个角被磨圆了的矩形,或者干脆是椭圆和圆.你甚至可以剪切任意的path,剪切使用addClip方法,在剪切了之后你可以针对剪切的这部分进行操作,例如你正在绘制一个卡片,这个卡片有小小的圆角效果,你可以把卡片绘制在一个矩形里面,然后把它剪切到一个小一点的圆角矩阵里,这样四角就修圆了.这是一个获得圆角卡片最简单的方法. 你也可以进行碰撞检测,它基于绕圈

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记37 TableView Delegate

上一话介绍了tableView的datasource,本话来介绍另一个重要的部分delegate. 当我们点击一个cell的时候,如何跳转到另外一个mvc中呢? 像增加其他segue一样,点击cell按住control键,右键连线到另一个mvc上,然后松手,选择需要的segue类型. 如果你的cell上还有其他按钮,比如detail disclosure,你也可以选择它的segue: 然后设置你的segue: 接着去prepareForSegue中设置这个segue: 每一个case对应不同的i

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记3 Xcode、Auto Layout及MVC

继续上一话中的计算器Demo.上一话讲到类必须被初始化.类中的属性也必须被初始化,所以你不能仅仅声明而不给它一个处置,那么问题来了,我们从storyboard中拖拽的@IBOutlet为什么仅仅有声明而不须要初始化呢,这是由于它的类型依然是一个optional,在你初始化之前已经被赋值为nil了,这也就是为什么你不须要再初始化它的原因. @IBOutlet weak var display: UILabel! 然而既然它是一个optional类型的,那为什么UILabel后面是"!"而

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记39 Alert&ActionSheet

Alert和ActionSheet是IOS中弹出消息的两个工具. 首先它们都是Modal的方式展示的. Alert用来向用户发起询问,可以有一个(比如取消)或两个选项(比如确定和取消),也可以附带一个文本框(比如要求用户输入密码) Action Sheet从屏幕底部滑出,提供一些分支的选项,选项的数量可以大于两个. 对比如图: Action Sheet和Alert都可以使用UIAlertController来创建,比如创建一个Action Sheet,在构造器中制定它的title和简介,注意St

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记23 多MVC模式Demo的实现

上一话我们对Demo的选择界面做了自动布局的相关处理,现在开始连接多个MVC的操作.首先我们需要其他工程中的文件,那么让我们打开另一个app.点击下面这个文件 然后拖动我们需要的文件到新的工程目录下: 注意勾选第一行,不然只是做了引用,如果你不小心删除了目标目录的话,你就找不到这些文件了,所以还是推荐做复制,这样会把文件复制到我们自己的工程目录下. 那么storyboard中的内容怎么办呢,我们只需要在原工程的storyboard中选中控制器然后复制,粘贴到新工程的storyboard中即可.

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记6 init

这一话首先来讲写关于init的东西. 首先初始化并不会经常被用到,这是因为类和结构体中的大部分属性都会通过赋值被初始化,或者有些属性是Optional的,这样即使是nil也没关系,可以在之后再给它们赋值,就好比StoryBoard中的outlet,又或者可以使用闭包来初始化,或者使用lazy来避开init,所以有很多方法来避免init,除非你确实需要一个init的时候,那么该怎么做呢? 在一些情况下会自动生成init,其中一种情况是当类中的所有属性都有初始值的时候,你会自动得到一个没有参数的初始

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记46 Persistence持久化

本话将介绍IOS中的四种数据持久化方式: Archiving SQLite File System Core Data 前面我们将结果NSUsrDefaults的用法,它是针对小数据量的持久化技术,本话的四种方式是针对大数据量的操作. 1.Archiving Archiving是一种把对象存储到硬盘上的存储方式,被存储的对象不需要所有属性的目录.只要一个对象实现了Archiving的两个方法,那么使用Archiving就可以存储这个对象的所有分支属性: func encodeWithCoder(