GCD练习

作者: 人生路02 | 来源:发表于2016-06-24 09:04 被阅读29次

GCD练习

ios 多线程 GCD : ios 多线程


  1. 全剧队列,异步执行
  2. 线程间通信
  3. 信号量
  4. 文件锁
  5. 单利模式
  6. 延时执行
  7. 取消任务
  8. 多核心遍历数组
  9. 队列组
  10. 消息传递机制

0.关于队列

1).创建或获取队列

// 获取全剧队列,并发队列
//第一个参数为:优先级,第二个参数为:苹果保留参数,写0即可。
#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
dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
// 获取主队列,串行队列
dispatch_queue_t queue = dispatch_get_main_queue();
    
// 创建队列 串行队列(DISPATCH_QUEUE_SERIAL 或者 NULL),并发队列(DISPATCH_QUEUE_CONCURRENT)
// 第一个参数为:队列名字,第二个参数为:队列类型,如上
dispatch_queue_t queue = dispatch_queue_create("name", DISPATCH_QUEUE_CONCURRENT)

1.全剧队列,异步执行

- (void)test7{
 
    dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
    for (int i = 0 ; i < 20; i++) {
        dispatch_async(queue, ^{
            NSLog(@"%@ - %d",[NSThread currentThread],i);
        });
    }
    dispatch_suspend(queue);
}

2. 线程间通信

- (void)gcd{
  
    dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
    
    for (int i =0 ; i<self.dataArray.count; i++) {
        
        // 异步添加操作到,全剧队列
        dispatch_async(queue, ^{
            
            NSData *data = [NSData dataWithContentsOfURL:[NSURL URLWithString:self.dataArray[i]]];
            UIImage *img = [UIImage imageWithData:data];
            
            UIImageView *imgView = (UIImageView *)_imgViewArray[i];
            imgView.image = [UIImage new];
            
            // 回到主线程更新UI
            dispatch_async(dispatch_get_main_queue(), ^{
                imgView.image = img;
            });
            
        });
    }
}

3. 信号量

1).信号量控制线程最大并发数量

- (void)test4{
    dispatch_semaphore_t semaphore = dispatch_semaphore_create(10);//为了让一次输出10个,初始信号量为10
    
    dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
    
    for (int i = 0; i <100; i++)
    {
        dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);//每进来1次,信号量-1;进来10次后就一直hold住,直到信号量大于0;
        dispatch_async(queue, ^{
            NSLog(@"%i - %@",i,[NSThread currentThread]);
            sleep(2);
            dispatch_semaphore_signal(semaphore);//由于这里只是log,所以处理速度非常快,我就模拟2秒后信号量+1;
        });
    }
    
}

4. 文件锁

- (void)test7{
// 必须传DISPATCH_QUEUE_CONCURRENT,不然和dispatch_async一样
    dispatch_queue_t queue = dispatch_queue_create("kk", DISPATCH_QUEUE_CONCURRENT);
//    dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
    // 这里可以是异步读文件
    for (int i = 0; i<10; i++) {
        dispatch_async(queue, ^{
            NSLog(@"111 - %@ - %d",[NSThread currentThread],i);
        });
    }
    // 只允许一个线程在,一个时间,修改数据
    dispatch_barrier_async(queue, ^{
        NSLog(@"%@ - 2222",[NSThread currentThread]);
    });
    
    // 其他线程异步读取,修改后的数据
    for (int i = 0; i<10; i++) {
        dispatch_async(queue, ^{
            NSLog(@"333 - %@ - %d",[NSThread currentThread],i);
        });
    }
}

5. 单利模式

static SingletonTimer * instance;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
    instance = [[SingletonTimer alloc] init];
});

return instance;

6. 延时执行

- (void)test7{
    dispatch_time_t time = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0 * NSEC_PER_SEC));
    dispatch_queue_t queue = dispatch_get_main_queue();
    dispatch_after(time, queue, ^{
        NSLog(@"%@",[NSThread currentThread]);
    });
}

7. 取消任务

- (void)test8{
    
    dispatch_queue_t concurrentDispatchQueue = dispatch_queue_create("com.test.queue", DISPATCH_QUEUE_CONCURRENT);
    
    dispatch_async(concurrentDispatchQueue, ^{
        for (int i=0; i<100; i++)
        {
            NSLog(@"%i",i);
            if (i==50)
            {
                NSLog(@"-----------------------------------");
                dispatch_suspend(concurrentDispatchQueue);
                sleep(3);
                dispatch_async(dispatch_get_main_queue(), ^{
                    dispatch_resume(concurrentDispatchQueue);
                });
            }
        }
    });
}
// 我们甚至可以在不同的线程对这个队列进行挂起和恢复,因为GCD是对队列的管理。

8. 多核心遍历数组

- (void)test11{

    NSArray *arr = [[NSArray alloc] initWithObjects:@"1",@"2",@"3", nil];
    
    dispatch_apply(arr.count, dispatch_get_global_queue(0, 0), ^(size_t i) {
        NSLog(@"%@ - %zu",[NSThread currentThread],i);
    });
    
    NSLog(@"over %@",[NSThread currentThread]);

}

9. 队列组

- (void)test9{
    
//    dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
    dispatch_group_t group = dispatch_group_create();
    
    dispatch_queue_t queueAll1 = dispatch_queue_create("hahaha1", DISPATCH_QUEUE_CONCURRENT);
    
    dispatch_queue_t queueAll2 = dispatch_queue_create("hahaha2", DISPATCH_QUEUE_CONCURRENT);
    
    dispatch_group_async(group, queueAll1, ^{
       
        for (int i =0 ; i<10; i++) {
            NSLog(@"%s - %d - %@",dispatch_queue_get_label(queueAll1),i,[NSThread currentThread]);
        }
        
    });
    
    dispatch_group_async(group, queueAll2, ^{
        for (int i =0 ; i<10; i++) {
            [NSThread sleepForTimeInterval:1.0];
            NSLog(@"%s - %d - %@",dispatch_queue_get_label(queueAll2),i,[NSThread currentThread]);
        }
    });
    
    dispatch_group_notify(group, dispatch_get_main_queue(), ^{
        for (int i =0 ; i<10; i++) {
            NSLog(@"%s - %d - %@",dispatch_queue_get_label(dispatch_get_main_queue()),i,[NSThread currentThread]);
        }
    });
    
}

10. 消息传递机制

- (void)test12{
    //创建一个数据对象,DISPATCH_SOURCE_TYPE_DATA_ADD的含义表示数据变化时相加
    source = dispatch_source_create(DISPATCH_SOURCE_TYPE_DATA_ADD, 0, 0, dispatch_get_main_queue());
    //创建接收数据变化的句柄
    dispatch_source_set_event_handler(source, ^{
        NSLog(@"%lu",dispatch_source_get_data(source));
    });
    //启动
    dispatch_resume(source);
    //设置数据
    dispatch_source_merge_data(source, 2);
    //这步执行完之后会执行打印方法
    
}

相关文章

  • GCD练习

    GCD练习 ios 多线程 GCD : ios 多线程 全剧队列,异步执行 线程间通信 信号量 文件锁 单利模式 ...

  • GCD学习总结

    GCD概述: OC: 相关练习: Swift 3.0: 相关练习: 参考资料:http://www.jianshu...

  • 多线程之GCD

    GCD介绍 1、GCD简介 2、GCD任务和队列 3、GCD 的基本使用 4、GCD 线程间的通信 5、GCD 的...

  • 扩展GCD(求逆元,解同余方程等等)

    首先要知道gcd函数的基本性质:gcd(a,b)=gcd(b,a)=gcd(|a|,|b|)=gcd(b,a%b)...

  • GCD死锁之探究

    先看代码 控制台只会打印 2018-01-03 16:35:01.315736+0800 GCD练习[3327:1...

  • iOS 一篇文章搞定GCD

    想了解NSOperation与GCD的区别可参考iOS多线程之NSOperation及简单练习 文章内容较长,介绍...

  • Swift 3.0 GCD写法练习

    demo

  • iOS - GCD

    目录 GCD简介 GCD核心概念 GCD队列的使用 GCD的常见面试题 GCD简介 Grand Central D...

  • iOS-多线程:GCD

    GCD 简介 GCD 任务和队列 GCD 的使用步骤 GCD 的基本使用(6种不同组合区别) GCD 线程间的通信...

  • 浅析GCD

    GCD目录: 1. GCD简介 为什么要用GCD呢? GCD可用于多核的并行运算GCD会自动利用更多的CPU内核(...

网友评论

      本文标题:GCD练习

      本文链接:https://www.haomeiwen.com/subject/oteodttx.html