iOS-多线程之GCD(原创)

前言

GCD

全称 Grand Central DisPath NSOperation便是基于GCD的封装

基础知识

1.GCD的优势

(1)为多核的并行运算提出了解决方案

(2)GCD会自动利用更多的CPU内核 比和双核 四核

(3).GCD自动管理线程的生命周期(创建线程 调度任务 销毁线程)

(4).程序员只需告诉GCD想要执行什么任务 不需要编写任何线程管理代码

2.GCD中有2个核心概念

任务: 执行什么操作

队列: 用来存放任务

3.队列可以分为两大类型

串行队列(Serial Dispatch Queue):只有一个线程,加入到队列中的操作按添加顺序依次执行,一个任务执行完毕后,才能再执行下一个任务。

并发队列(Concurrent Dispatch Queue):有多个线程,操作进来以后他会将这些线程安排在可用的处理器上,同时保证先进来的任务优先处理。

其实在GCD中还有一个特殊队列就是主队列 主队列中永远只有一个线程-主线程 用来执行主线程的操作任务

4.采用GCD做多线程 可以抽象分为二步

(1)找到队列(主队列或串行队列或并行队列)

(2)在队列中用同步或者异步的方式执行任务

5.执行队列中的任务的二种方式

(1)同步 只能在当前线程执行任务 不具备开启新线程的能力--主线程

(2)异步 可以在新的线程中执行任务 具备开启新线程的能力--子线程

下面介绍一下串行 并行 同步 异步

- (void)viewDidLoad {

[super viewDidLoad];

self.view.backgroundColor = [UIColor whiteColor];

// Do any additional setup after loading the view, typically from a nib.

#pragma mark ====串行同步====

//    //1.找到队列 第一个参数:该队列的名字 第二个参数:指定队列的类型

//    dispatch_queue_t serialQueue = dispatch_queue_create("serialQueue",DISPATCH_QUEUE_SERIAL);

//    //2.给队列指定任务 第一个参数:任务在哪个队列中执行 第二个参数:想要执行的操作

//    //asyn是异步 syn是同步

//    dispatch_sync(serialQueue, ^{

//        NSLog(@"1===%@",[NSThread currentThread]);

//    });

//

#pragma mark ====串行异步====

//    dispatch_queue_t serialQueue = dispatch_queue_create("serialQueue",DISPATCH_QUEUE_SERIAL);

//    dispatch_async(serialQueue, ^{

//        NSLog(@"1===%@",[NSThread currentThread]);

//    });

#pragma mark ====并行同步====

//    dispatch_queue_t concurrentQueue = dispatch_queue_create("concurrentQueue",DISPATCH_QUEUE_CONCURRENT);

//    dispatch_sync(concurrentQueue, ^{

//        NSLog(@"1===%@",[NSThread currentThread]);

//    });

#pragma mark ====并行异步====

dispatch_queue_t concurrentQueue = dispatch_queue_create("concurrentQueue",DISPATCH_QUEUE_CONCURRENT);

dispatch_async(concurrentQueue, ^{

NSLog(@"1===%@",[NSThread currentThread]);

});

}

此时我们新建一个类来看一下 用GCD的形式来加载网络图片让它显示在self.view上 我这里为它命名为OneImageViewController  效果图以及.m代码如下

#import "OneImageViewController.h"
#define kurl @"http://store.storeimages.cdn-apple.com/8748/as-images.apple.com/is/image/AppleInc/aos/published/images/s/38/s38ga/rdgd/s38ga-rdgd-sel-201601?wid=848&hei=848&fmt=jpeg&qlt=80&op_sharpen=0&resMode=bicub&op_usm=0.5,0.5,0,0&iccEmbed=0&layer=comp&.v=1454777389943"
@interface OneImageViewController ()
{
    UIImageView *imageView;
}
@end

@implementation OneImageViewController
- (void)viewDidLoad {
    [super viewDidLoad];
    self.view.backgroundColor = [UIColor whiteColor];
    /*
     1.创建视图
     2.创建一个串行队列
     3.用异步方式执行队列中的任务
     4.加载网络资源
     5.回到主线程 更新UI

     */
    //1.创建视图
    imageView = [[UIImageView alloc]initWithFrame:CGRectMake(50, 50, 200, 200)];
    [self.view addSubview:imageView];
    //2.创建一个串行队列
    dispatch_queue_t serialQueue = dispatch_queue_create("serialQueue", DISPATCH_QUEUE_SERIAL);
    //3.用异步方式执行队列中的任务
    dispatch_async(serialQueue, ^{
        //4.加载网络资源
        NSData *data = [NSData dataWithContentsOfURL:[NSURL URLWithString:kurl]];
        UIImage *image = [UIImage imageWithData:data];
        //5.回到主线程 dispatch_get_main_queue这个函数 找到主队列
        dispatch_queue_t mainQueue = dispatch_get_main_queue();
        dispatch_sync(mainQueue, ^{
           //6.更新UI
            imageView.image = image;
        });

    });

}
@end

  

利用GCD加载多张网络图片 我在这里给类命名为MoreImageViewViewController 效果图以及.m代码如下

#import "MoreImageViewViewController.h"
#define kurl @"http://store.storeimages.cdn-apple.com/8748/as-images.apple.com/is/image/AppleInc/aos/published/images/s/38/s38ga/rdgd/s38ga-rdgd-sel-201601?wid=848&hei=848&fmt=jpeg&qlt=80&op_sharpen=0&resMode=bicub&op_usm=0.5,0.5,0,0&iccEmbed=0&layer=comp&.v=1454777389943"
@interface MoreImageViewViewController ()
{
    int imageIndex;
    dispatch_queue_t concurrentQueue;

}
@end

@implementation MoreImageViewViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    self.view.backgroundColor = [UIColor whiteColor];
    self.edgesForExtendedLayout = UIRectEdgeNone;
    /*
     1.创建多个视图
     2.找到并行队列
     3.给这个并行队列指定多个任务
     4.在子线程加载网络资源
     5.回到主线程
     6.更新UI
     */

    imageIndex = 100;

    //1.创建多个视图
    for (int row = 0; row<3; row++) {
        for (int list = 0; list<2; list++) {

            UIImageView *imageView = [[UIImageView alloc]initWithFrame:CGRectMake(10+list*200, 10+row*200, 180, 180)];

            //imageView.backgroundColor = [UIColor orangeColor];

             imageView.tag = imageIndex++;

            [self.view addSubview:imageView];

        }
    }
    //2.找到并行队列 dispatch_get_global_queue 获取到系统的全局并列队列

    //第一个参数:是优先级 第二个参数:保留参数 没用
//    dispatch_queue_t concurrentQueue = dispatch_get_global_queue(0, 0);

      concurrentQueue = dispatch_queue_create("concurrentQueue", DISPATCH_QUEUE_SERIAL);

    //3.给这个并行队列指定多个任务
    for (int index = 0; index<6; index++) {
        dispatch_async(concurrentQueue, ^{
            [NSThread sleepForTimeInterval:0.5];
            //4.加载网络资源
            NSData *data = [NSData dataWithContentsOfURL:[NSURL URLWithString:kurl]];
            UIImage *image = [UIImage imageWithData:data];
            //5.回到主线程
            dispatch_sync(dispatch_get_main_queue(), ^{
                //6.更新UI
                UIImageView *imageView = [self.view viewWithTag:100+index];
                imageView.image = image;
            });

        });
    }

    [self controlBtn];
}

- (void)controlBtn{

    UISegmentedControl *segment = [[UISegmentedControl alloc]initWithItems:@[@"暂停",@"开启",]];

    segment.frame = CGRectMake(50, 620, 300, 50);

    segment.apportionsSegmentWidthsByContent = YES;

    [self.view addSubview:segment];

    [segment addTarget:self action:@selector(clickSegment:) forControlEvents:UIControlEventValueChanged];
}

- (void)clickSegment:(UISegmentedControl *)sender {

    switch (sender.selectedSegmentIndex) {

        case 0:{
            //暂停队列
            dispatch_suspend(concurrentQueue);
        }break;

        case 1:{
            //恢复队列
            dispatch_resume(concurrentQueue);

        }break;

    }

开发中我们可能会用到线程锁 比如购票抢票这一功能

没线程锁的情况下: 我走进购票大厅,买票的人都没有排队,我好不容易挤到窗口前,正打算掏钱买票的时候,旁边有人已经把钱给了售票员。虽然你的线程已经开始执行买票的方法,但当你去拿票时,也就是将票数减一时,CPU将你的线程给中断,开始执行其他的线程,CPU返回继续执行你的线程的时候,票已经没了。

有线程锁的情况下:* 我走进购票大厅,买票的人都在排队,当我到柜台能保证我买票的关键过程,也就是报站、掏钱、拿票过程不受干扰,我采用线程锁将这个关键过程给锁起来,以保证我能顺利的买到票。

我在这里命名为GCDLockViewController 具体.m代码如下

#import "GCDLockViewController.h"

@interface GCDLockViewController ()
{
    NSLock *mylock;
}
@end

@implementation GCDLockViewController
- (void)viewDidLoad {

    [super viewDidLoad];

    self.view.backgroundColor = [UIColor whiteColor];
    //实例化一个线程锁
    mylock = [NSLock new];
#pragma mark ====线程锁====
    __block int ticketNum = 10;
    dispatch_queue_t concurrent = dispatch_get_global_queue(0, 0);
    for (int index = 0; index<15; index++) {
        dispatch_async(concurrent, ^{

//            [mylock lock];
//            if (ticketNum>0) {
//                ticketNum--;
//                NSLog(@"还剩%d张票",ticketNum);
//            }
//            [mylock unlock];
            //参数一般是self 与self相关的变量 多个线程同时同时只访问一次
            @synchronized(self) {
                if (ticketNum>0) {
                                    ticketNum--;
                                    NSLog(@"还剩%d张票",ticketNum);
                                }

            }

        });
    }

}
@end

 

时间: 2025-01-11 22:11:51

iOS-多线程之GCD(原创)的相关文章

iOS多线程之GCD小记

iOS多线程之GCD小记 iOS多线程方案简介 从各种资料中了解到,iOS中目前有4套多线程的方案,分别是下列4中: 1.Pthreads 这是一套可以在很多操作系统上通用的多线程API,是基于C语言的,在在oc中使用时需要包含 #import<pthread.h> 使用这种多线程方案需要手动处理线程的各个状态的转换,也就是要管理线程的生命周期. 2.NSThread 这种多线程方案经过了苹果的封装,是一种面向对象的方案,因此可以直接操控线程对象,相对来说比较便捷,其生命周期也要手动管理 3.

(五十五)iOS多线程之GCD

GCD的全称为Grand Central Dispatch,翻译为大中央调度,是Apple开发的一个多线程编程解决方法. 进程和线程的概念: 正在进行中的程序被称为进程,负责程序运行的内存分配,每一个进程都有自己独立的虚拟内存空间. 线程是进程中一个独立的执行路径,即主线程,主线程有1M的栈区,对于耗时的执行路径,可以放在子线程(512K栈区)中执行. Tip:新建线程会消耗内存空间和CPU事件,线程太多会降低系统的运行性能,多线程是通过CPU时分复用实现的. Tip:多线程是为了并发执行多项任

iOS多线程之GCD学习笔记

什么是GCD 1.全称是Grand Central Dispatch,可译为“牛逼的中枢调度器” 2.纯C语言,提供了非常多强大的函数 GCD的优势 GCD是苹果公司为多核的并行运算提出的解决方案 GCD会自动利用更多的CPU内核(比如双核.四核) GCD会自动管理线程的生命周期(创建线程.调度任务.销毁线程) 程序员只需要告诉GCD想要执行什么任务,不需要编写任何线程管理代码 任务和队列 GCD中有2个核心概念 任务:执行什么操作 队列:用来存放任务 GCD的使用就2个步骤,首先确定定制任务(

iOS多线程之GCD的基本使用

使用GCD开发的基本思路 基本思想:将任务(操作)放在队列中去执行 任务使用block定义 队列负责调度任务执行所在的线程以及具体的执行时间 队列的特点是先进先出(FIFO)的,新添加至队列的操作都会排在队尾. GCD的函数都是以dispatch开头的 注意 :队列不是线程,也不对应CPU. 队列:dispatch_queue_t 两种队列:串行队列.并行队列 队列上的操作:添加任务 两种任务:同步任务.异步任务 队列及任务 GCD的串行队列,意味着队列中的任务排队执行 1)添加异步任务:创建一

iOS 多线程之GCD的使用

在iOS开发中,遇到耗时操作,我们经常用到多线程技术.Grand Central Dispatch (GCD)是Apple开发的一个多核编程的解决方法,只需定义想要执行的任务,然后添加到适当的调度队列(dispatch queue).GCD会负责创建线程和调度你的任务,系统直接提供线程管理. 一.队列: 基本概念: 1.GCD的一个重要概念是队列,它的核心理念:将长期运行的任务拆分成多个工作单元,并将这些单元添加到dispath queue中,系统会为我们管理这些dispath queue,为我

ios多线程之GCD

** dispatch_after 延时操作应用场景 例如:游戏后台需要做一些随机的事件,需要在某个时间后,调用方法! 1> 调用的方法通常是跟UI有关的,例如提示用户等 2> 不了解GCD或者多线程的人,可以直接填空即可 */ - (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event { [self delay1]; } #pragma mark - 延时操作 /** 在其他线程中调用 dispatch_after */

Objective-C IOS多线程之GCD深入理解

在 GCD 中,加入了两个非常重要的概念:任务和队列一个线程是可以拥有多个执行队列的,所有任务是添加到队列中等待执行的主队列是特殊的串行队列,自己创建的队列可以指定串行或并行,全局队列是并行队列 任务:即操作,你想要干什么,说白了就是一段代码,在 GCD 中就是一个 Block,所以添加任务十分方便.任务有两种执行方式: 同步执行和异步执行,他们之间的区别主要在于会不会阻塞当前线程 首先看下面这两个例子: 1.dispatch_queue_t queue = dispatch_queue_cre

OC多线程之GCD

要了解多线程首先要知道什么是进程,什么是进程? 正在进行中的程序被称为进程,负责程序运行的内存分配 每一个进程都有自己独立的虚拟内存空间 什么是线程: 线程是进程中一个独立的执行路径(控制单元) 一个进程中至少包含一条线程,即主线程 可以将耗时的执行路径(如:网络请求)放在其他线程中执行 创建线程的目的就是为了开启一条新的执行路径,运行指定的代码,与主线程中的代码实现同时运行 线程的优缺点: 优势 (1)充分发挥多核处理器优势,将不同线程任务分配给不同的处理器,真正进入“并行运算”状态 (2)将

OC多线程之GCD ----- 2

dispatch_create生成的Queue不管是并行队列还是串行队列,其优先级都是默认优先级 但是可以用dispatch_set_target_queue来改变队列的优先级 dispatch_set_target_queue(原来的队列, 目标优先级队列) 使用这个函数需要获取两个队列,一个是需要变更优先级的队列,一个是指定优先级的队列(指定优先级的队列可以通过get_global获得) 如果多个串行队列优先级相同,那么这些队列里的任务也会串行执行 dispatch_after函数并不能非常