iOS:多线程技术GCD的使用

简介:

GCD的使用:

1.队列的类型
     1.1 主队列:mian queue,主线程队列,负责更行UI的操作。是一个串行的队列。
     1.2 系统默认的并行队列:global queue,按优先级分类。
     1.3 自定义的队列:可以创建串行队列或者是并行的队列
 
2.任务
     2.1 封装的形式:block方法或C语言的函数
     2.2 添加到队列的方式:同步或异步(只对并行队列有区别)
         例如服务器请求:
         同步:提交请求->等待服务器处理(这个期间客户端浏览器不能干任何事)->处理完毕返回   
         异步:请求通过事件触发->服务器处理(这个期间客户端浏览器仍可以干其他的事)->处理完毕返回
 
3.特殊使用
     3.1 只执行一次 dispatch_once(dispatch_once_t *predicate, dispatch_block_t block);(多用于单例模式)
     3.2 延时执行    dispatch_after(dispatch_time_t when,dispatch_queue_t  queue,dispatch_block_t block);
     3.3 成组的执行任务       dispatch_group_create(void);
     3.4 创建自定义的队列    dispatch_queue_create(const char *label, dispatch_queue_attr_t attr);
     3.5 创建默认的全局队列 dispatch_get_global_queue(long identifier, unsigned long flags)
     3.6 获取主队列             dispatch_get_main_queue(void)
     3.7 异步执行                 dispatch_async(dispatch_queue_t queue, dispatch_block_t block);
     3.8 同步执行              dispatch_sync(dispatch_queue_t queue, dispatch_block_t block);
     3.9 多次执行  dispatch_apply(size_t iterations, dispatch_queue_t queue,void (^block)(size_t));
     …………………等等…………………
 

4.几个方法参数解释:

<1>创建自定义的队列    dispatch_queue_create(const char *label, dispatch_queue_attr_t attr);

const char *label:队列名字                   dispatch_queue_attr_t attr:队列执行方式(串行、并行)

<2>创建默认的全局队列 dispatch_get_global_queue(long identifier, unsigned long flags)

long identifier:队列执行的优先级           unsigned long flags:默认为0即可

<3>多次执行         dispatch_apply(size_t iterations, dispatch_queue_t queue,void (^block)(size_t));

size_t iterations:执行次数      dispatch_queue_t queue:队列      void (^block)(size_t):block函数块

 
5.队列方式宏定义(用于创建自定义队列时的参数)

#define DISPATCH_QUEUE_SERIAL NULL         //串行

#define DISPATCH_QUEUE_CONCURRENT        //并行

 

6.队列优先级宏定义(创建全局队列时的参数)

#define DISPATCH_QUEUE_PRIORITY_HIGH 2

#define DISPATCH_QUEUE_PRIORITY_DEFAULT 0

#define DISPATCH_QUEUE_PRIORITY_LOW (-2)

#define DISPATCH_QUEUE_PRIORITY_BACKGROUND INT16_MIN

 

7.更能区分

主队列:专门用来执行主线程的,进行UI的更新操作

全局队列或自定义队列:用来添加并执行其他的线程,进行数据的操作

 

 具体举例如下:

例子1:采用不分组的方式,将多线程添加到队列中,然后进行多线程的操作。

1.准备UI界面布局:拖入一个文本视图控件,并关联相关的类中,同时在类中声明一个票数变量

复制代码
@interface ViewController ()
{
    NSInteger _tickets;
}
@property (weak, nonatomic) IBOutlet UITextView *textView;
@end
复制代码

2.设置票数,同时将原来文本视图中默认的数据清空,取消自动布局便于后面添加数据时自动滚动文本视图

 //设置数据和文本视图
 _tickets = 20;
 [self.textView setText:@""];
 self.textView.layoutManager.allowsNonContiguousLayout = NO;

3.创建全局队列,并设置优先级,设置并行方式

dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

4.用GCD创建任务线程,并将线程添加到队列中,采用异步执行方式

复制代码
dispatch_async(queue, ^{
        [self GCDSellTicketMethod:@"GCD售票线程-1"];
    });
    
dispatch_async(queue, ^{
        [self GCDSellTicketMethod:@"GCD售票线程-2"];
    });
复制代码

5.定义更新UI的方法

#pragma mark -更新UI的操作

复制代码
-(void)appendTextView:(NSString *)text
{
    //1.获取原来的数据
    NSMutableString *content = [NSMutableString stringWithString:self.textView.text];
    NSRange range = NSMakeRange(content.length, 2);

    //2.追加新的内容
    [content appendString:[NSString stringWithFormat:@"\n%@",text]];
    [self.textView setText:content];
    
    //3.滚动视图
    [self.textView scrollRangeToVisible:range];
}
复制代码

6.定义任务线程的执行方法

#pragma mark -执行线程的操作

复制代码
-(void)GCDSellTicketMethod:(NSString *)name
{
    while (YES)
    {
        if(_tickets > 0)
        {
            //使用GCD
            dispatch_async(dispatch_get_main_queue(), ^{
                //更新UI
                NSString *info = [NSString stringWithFormat:@"总票数:%ld,当前的线程:%@",_tickets,name];
                [self appendTextView:info];
                
                //卖票
                _tickets--;
            });
            
            //线程休眠
            if([name isEqualToString:@"GCD售票线程-1"])
            {
                [NSThread sleepForTimeInterval:0.3f];
            }
            else
            {
                [NSThread sleepForTimeInterval:0.2f];
            }
        }
        else
        {
            //使用GCD更新UI
            dispatch_async(dispatch_get_main_queue(), ^{
                NSString *info = [NSString stringWithFormat:@"票已经卖完,当前线程:%@",name];
                [self appendTextView:info];
            });
            
            //退出线程
            break;
        }
    }
}
复制代码

 

例子2:采用分组的方式,将线程组添加到队列中,然后进行多线程的操作。

1.准备UI界面布局:拖入一个文本视图控件,并关联相关的类中,同时在类中声明一个票数变量

复制代码
@interface ViewController ()
{
    NSInteger _tickets;
}
@property (weak, nonatomic) IBOutlet UITextView *textView;
@end
复制代码

2.设置票数,同时将原来文本视图中默认的数据清空,取消自动布局便于后面添加数据时自动滚动文本视图

 //设置数据和文本视图
 _tickets = 20;
 [self.textView setText:@""];
 self.textView.layoutManager.allowsNonContiguousLayout = NO;

3.创建一个线程分组

dispatch_group_t group = dispatch_group_create();

4.创建自定义的队列,并设置队列执行方式为并行方式

  dispatch_queue_t queue = dispatch_queue_create("myQueue", DISPATCH_QUEUE_CONCURRENT);

5.用GCD创建任务线程组,并将线程组添加到队列中,采用分组异步执行方式

复制代码

  dispatch_group_async(group,queue, ^{

      [self GCDSellTicketMethod:@"GCD售票线程-1"];

  });   

  dispatch_group_async(group,queue, ^{

      [self GCDSellTicketMethod:@"GCD售票线程-2"];

  });

复制代码

6.等线程组中的所有任务完成后,会接收到通知,更新UI

复制代码
dispatch_group_notify(group, queue, ^{
        dispatch_async(dispatch_get_main_queue(), ^{
            NSString *info = [NSString stringWithFormat:@"票已经卖完"];
            [self appendTextView:info];
        });
    });
复制代码

7.定义更新UI的方法

#pragma mark -更新UI的操作

复制代码
-(void)appendTextView:(NSString *)text
{
    //1.获取原来的数据
    NSMutableString *content = [NSMutableString stringWithString:self.textView.text];
    NSRange range = NSMakeRange(content.length, 2);

    //2.追加新的内容
    [content appendString:[NSString stringWithFormat:@"\n%@",text]];
    [self.textView setText:content];
    
    //3.滚动视图
    [self.textView scrollRangeToVisible:range];
}
复制代码

8.定义任务线程的执行方法

#pragma mark -执行线程的操作

复制代码
-(void)GCDSellTicketMethod:(NSString *)name
{
    while (YES)
    {
        if(_tickets > 0)
        {
            //使用GCD
            dispatch_async(dispatch_get_main_queue(), ^{
                //更新UI
                NSString *info = [NSString stringWithFormat:@"总票数:%ld,当前的线程:%@",_tickets,name];
                [self appendTextView:info];
                
                //卖票
                _tickets--;
            });
            
            //线程休眠
            if([name isEqualToString:@"GCD售票线程-1"])
            {
                [NSThread sleepForTimeInterval:0.3f];
            }
            else
            {
                [NSThread sleepForTimeInterval:0.2f];
            }
        }
        else
        {   
            //退出线程
            break;
        }
    }
}
复制代码

 

两种情况的演示结果如下:

 

 

 

 

 

 

 

程序猿神奇的手,每时每刻,这双手都在改变着世界的交互方式!
分类:  iOS高级

本文转自当天真遇到现实博客园博客,原文链接:http://www.cnblogs.com/XYQ-208910/p/4859089.html,如需转载请自行联系原作者
相关文章
|
11天前
|
安全 Android开发 iOS开发
安卓与iOS的较量:技术特性与用户体验的深度解析
在移动操作系统的战场上,安卓和iOS一直占据着主导地位。本文将深入探讨这两大平台的核心技术特性,以及它们如何影响用户的体验。我们将从系统架构、应用生态、安全性能和创新功能四个方面进行比较,帮助读者更好地理解这两个系统的异同。
41 3
|
17天前
|
物联网 区块链 vr&ar
未来已来:探索区块链、物联网与虚拟现实技术的融合与应用安卓与iOS开发中的跨平台框架选择
【8月更文挑战第30天】在科技的巨轮下,新技术不断涌现,引领着社会进步。本文将聚焦于当前最前沿的技术——区块链、物联网和虚拟现实,探讨它们各自的发展趋势及其在未来可能的应用场景。我们将从这些技术的基本定义出发,逐步深入到它们的相互作用和集成应用,最后展望它们如何共同塑造一个全新的数字生态系统。
|
4天前
|
API Android开发 iOS开发
安卓与iOS开发中的线程管理对比
【9月更文挑战第12天】在移动应用的世界中,安卓和iOS平台各自拥有庞大的用户群体。开发者们在这两个平台上构建应用时,线程管理是他们必须面对的关键挑战之一。本文将深入探讨两大平台在线程管理方面的异同,通过直观的代码示例,揭示它们各自的设计理念和实现方式,帮助读者更好地理解如何在安卓与iOS开发中高效地处理多线程任务。
|
2天前
|
API Android开发 iOS开发
掌握安卓与iOS应用开发中的依赖注入技术
本文探讨了在安卓和iOS应用开发中,如何有效利用依赖注入技术来提升代码的模块化、可测试性和可维护性。通过对比分析两种平台下依赖注入的实现方式与工具,本文旨在为开发者提供一套清晰、实用的依赖管理策略,助力打造高质量软件产品。
|
10天前
|
安全 Android开发 iOS开发
安卓与iOS的较量:技术特性与用户体验的深度剖析
在移动操作系统的战场上,安卓和iOS一直是两个重量级选手。本文将深入探讨两者的技术架构、安全性、应用生态以及用户体验等方面的差异,并尝试从用户和开发者的角度出发,分析这两个系统的优势与不足。通过比较,我们不仅能更好地理解各自的特点,还能洞察未来移动技术的发展趋势。
|
13天前
|
监控 Java
线程池中线程异常后:销毁还是复用?技术深度剖析
在并发编程中,线程池作为一种高效利用系统资源的工具,被广泛用于处理大量并发任务。然而,当线程池中的线程在执行任务时遇到异常,如何妥善处理这些异常线程成为了一个值得深入探讨的话题。本文将围绕“线程池中线程异常后:销毁还是复用?”这一主题,分享一些实践经验和理论思考。
28 3
|
19天前
|
移动开发 搜索推荐 Android开发
安卓与iOS开发:一场跨平台的技术角逐
在移动开发的广阔舞台上,两大主角——安卓和iOS,持续上演着激烈的技术角逐。本文将深入浅出地探讨这两个平台的开发环境、工具和未来趋势,旨在为开发者揭示跨平台开发的秘密,同时激发读者对技术进步的思考和对未来的期待。
|
22天前
|
安全 Android开发 Swift
安卓与iOS开发:平台差异与技术选择
【8月更文挑战第26天】 在移动应用开发的广阔天地中,安卓和iOS两大平台各占一方。本文旨在探索这两个系统在开发过程中的不同之处,并分析开发者如何根据项目需求选择合适的技术栈。通过深入浅出的对比,我们将揭示各自平台的优势与挑战,帮助开发者做出更明智的决策。
|
1月前
|
API Windows
揭秘网络通信的魔法:Win32多线程技术如何让服务器化身超级英雄,同时与成千上万客户端对话!
【8月更文挑战第16天】在网络编程中,客户/服务器模型让客户端向服务器发送请求并接收响应。Win32 API支持在Windows上构建此类应用。首先要初始化网络环境并通过`socket`函数创建套接字。服务器需绑定地址和端口,使用`bind`和`listen`函数准备接收连接。对每个客户端调用`accept`函数并在新线程中处理。客户端则通过`connect`建立连接,双方可通过`send`和`recv`交换数据。多线程提升服务器处理能力,确保高效响应。
35 6
|
1月前
|
移动开发 开发工具 Android开发
探索安卓与iOS开发的差异:技术选择的影响
【8月更文挑战第17天】 在移动应用开发的广阔天地中,安卓和iOS两大平台各领风骚。本文通过比较这两个平台的编程语言、开发工具及市场策略,揭示了技术选择对开发者和产品成功的重要性。我们将从开发者的视角出发,深入探讨不同平台的技术特性及其对项目实施的具体影响,旨在为即将步入移动开发领域的新手提供一个清晰的指南,同时给予资深开发者新的思考角度。