GCD实例

转自http://blog.csdn.net/jjunjoe/article/details/8743434,学习了,感谢


示例源码清单如下:

1SingletonSample.h

//

//  SingletonSample.h

//  BlockSample

//

//  Created by developer on 13-9-27.

//  Copyright (c) 2013 developer. All rights reserved.

//


#import <Foundation/Foundation.h>


@interface SingletonSample : NSObject


+ (id)ShareInstance;

@end



2SingletonSample.m

//

//  SingletonSample.m

//  BlockSample

//

//  Created by developer on 13-9-27.

//  Copyright (c) 2013 developer. All rights reserved.

//


#import "SingletonSample.h"


@implementation SingletonSample

+ (id)ShareInstance{

    static SingletonSample *SharedInstance = nil;

    static dispatch_once_t onceToken;

    dispatch_once(&onceToken, ^{

        SharedInstance = [[SingletonSample alloc] init];

    });

    

    return SharedInstance;

}

@end


3BlockSampleTests.h

//

//  BlockSampleTests.h

//  BlockSampleTests

//

//  Created by developer on 13-8-3.

//  Copyright (c) 2013 developer. All rights reserved.

//


#import <SenTestingKit/SenTestingKit.h>


@interface BlockSampleTests : SenTestCase


@end


4BlockSampleTests.m

//

//  BlockSampleTests.m

//  BlockSampleTests

//

//  Created by developer on 13-8-3.

//  Copyright (c) 2013 developer. All rights reserved.

//


#import "SingletonSample.h"

#import "BlockSampleTests.h"


static int global = 100;


static volatile BOOL flag = NO;


static const int Length = 100;

static int data[Length];

static void initData()

{

    for(int i = 0; i < Length; i++)

        data[i] = i + 1;

}



@implementation BlockSampleTests


- (void)setUp

{

    [super setUp];

    

    // Set-up code here.

}


- (void)tearDown

{

    // Tear-down code here.

    

    [super tearDown];

}


- (void)test1

{

    // 默认打印一条语句"Hello, World!"

    NSLog(@"Hello, World!");

}


- (void)testBlock1

{

    // block 打印一条语句"Hello, World!"

    void (^aBlock)(void) = ^(void){ NSLog(@"Hello, World!"); };

    aBlock();

}


- (void)testBlock2

{

    // block 打印一条语句"Hello, World!"

    void (^aBlock)(void) = 0;

    aBlock = ^(void) {

        NSLog(@"Hello, World!");

    };

    

    aBlock();

}


- (void)testBlockArray

{

    // block 数组

    void (^blocks[2])(void) = {

        ^(void){ NSLog(@" >> This is block 1!"); },

        ^(void){ NSLog(@" >> This is block 2!"); }

    };

    

    blocks[0]();

    blocks[1]();

}


- (void)testBlock4

{

    /*

     block 是分配在 stack 上的,这意味着我们必须小心里处理 block 的生命周期。

     比如如下的做法是不对的,因为 stack 分配的 block  if  else 内是有效的,但是到大括号 } 退出时就可能无效了:

     */

    

    dispatch_block_t block;

    BOOL x = 1;

    

    if (x) {

        block = ^{ printf("true\n"); };

    } else {

        block = ^{ printf("false\n"); };

    }

    block();

}


- (void)testBlock5

{

    /*

     考虑到 block 的目的是为了支持并行编程,对于普通的 local 变量,我们就不能在 block 里面随意修改(原因很简单,block 可以被多个线程并行运行,会有问题的)

     而且如果你在 block 中修改普通的 local 变量,编译器也会报错。

     那么该如何修改外部变量呢?有两种办法:

     第一种是可以修改 static 全局变量;

     第二种是可以修改用新关键字 __block 修饰的变量。

     */

    __block int blockLocal  = 100;

    static int staticLocal  = 100;

    

    void (^aBlock)(void) = ^(void){

        NSLog(@" >> Sum: %d\n", global + staticLocal);

        

        global++;

        blockLocal++;

        staticLocal++;

    };

    

    aBlock();

    

    NSLog(@"After modified, global: %d, block local: %d, static local: %d\n", global, blockLocal, staticLocal);

    

}


- (void)testBlock6

{

    /*

     我们也可以引用 static block  __block block。比如我们可以用他们来实现 block 递归

     */

    

    void (^aBlock)(int) = 0;

    static void (^ const staticBlock)(int) = ^(int i) {

        if (i > 0) {

            NSLog(@" >> static %d", i);

            staticBlock(i - 1);

        }

    };

    

    aBlock = staticBlock;

    aBlock(5);

    

    // 2

    __block void (^blockBlock)(int);

    blockBlock = ^(int i) {

        if (i > 0) {

            NSLog(@" >> block %d", i);

            blockBlock(i - 1);

        }

    };

    

    blockBlock(5);

}


- (void)testBlock7

{

    /*

     上面我们介绍了 block 及其基本用法,但还没有涉及并行编程。

     block  Dispatch Queue 分发队列结合起来使用,是 iOS 中并行编程的利器。

     */

    

    // create dispatch queue

    dispatch_queue_t queue = dispatch_queue_create("StudyBlocks",NULL);

    

    dispatch_async(queue, ^(void) {

        int sum = 0;

        for(int i = 0; i < Length; i++)

            sum += data[i];

        

        NSLog(@" >> Sum: %d", sum);

        

        flag = YES;

    });

    

    // wait util work is done.

    while (!flag);

    dispatch_release(queue);

    

}


- (void)testBlock8

{

    /*

     在上面的例子中,我们的主线程一直在轮询 flag 以便知晓 block 线程是否执行完毕,这样做的效率是很低的,严重浪费 CPU 资源。

     我们可以使用一些通信机制来解决这个问题,如:semaphore(信号量)。

     semaphore 的原理很简单,就是生产-消费模式,必须生产一些资源才能消费,没有资源的时候,那我就啥也不干,直到资源就绪。

     */

    

    initData();

    

    // Create a semaphore with 0 resource

    __block dispatch_semaphore_t sem = dispatch_semaphore_create(0);

    

    // create dispatch semaphore

    dispatch_queue_t queue = dispatch_queue_create("StudyBlocks",NULL);

    

    dispatch_async(queue, ^(void) {

        int sum = 0;

        for(int i = 0; i < Length; i++)

            sum += data[i];

        

        NSLog(@" >> Sum: %d", sum);

        

        // signal the semaphore: add 1 resource

        dispatch_semaphore_signal(sem);

    });

    

    // wait for the semaphore: wait until resource is ready.

    dispatch_semaphore_wait(sem, DISPATCH_TIME_FOREVER);

    

    dispatch_release(sem);

    dispatch_release(queue);

}


- (void)testBlock9

{

    /*

     下面我们来看一个按照 FIFO 顺序执行并用 semaphore 同步的例子:先将数组求和再依次减去数组。

     */

    initData();

    

    __block int sum = 0;

    

    // Create a semaphore with 0 resource

    __block dispatch_semaphore_t sem = dispatch_semaphore_create(0);

    __block dispatch_semaphore_t taskSem = dispatch_semaphore_create(0);

    

    // create dispatch semaphore

    dispatch_queue_t queue = dispatch_queue_create("StudyBlocks",NULL);

    

    dispatch_block_t task1 = ^(void) {

        int s = 0;

        for (int i = 0; i < Length; i++)

            s += data[i];

        sum = s;

        

        NSLog(@" >> after add: %d", sum);

        

        dispatch_semaphore_signal(taskSem);

    };

    

    dispatch_block_t task2 = ^(void) {

        dispatch_semaphore_wait(taskSem, DISPATCH_TIME_FOREVER);

        

        int s = sum;

        for (int i = 0; i < Length; i++)

            s -= data[i];

        sum = s;

        

        NSLog(@" >> after subtract: %d", sum);

        dispatch_semaphore_signal(sem);

    };

    

    dispatch_async(queue, task1);

    dispatch_async(queue, task2);

    

    // wait for the semaphore: wait until resource is ready.

    dispatch_semaphore_wait(sem, DISPATCH_TIME_FOREVER);

    

    dispatch_release(taskSem);

    dispatch_release(sem);

    dispatch_release(queue);

    

    /*

     在上面的代码中,我们利用了 dispatch_queue  FIFO 特性,

     确保 task1 先于 task2 执行,而 task2 必须等待直到 task1 执行完毕才开始干正事,主线程又必须等待 task2 才能干正事。

     这样我们就可以保证先求和,再相减,然后再让主线程运行结束这个顺序。

     */

    

}


- (void)testBlock10

{

    /*

     使用 dispatch_apply 进行并发迭代:

     对于上面的求和操作,我们也可以使用 dispatch_apply 来简化代码的编写:

     */

    

    initData();

    

    dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

    

    __block int sum = 0;

    __block int *pArray = data;

    

    // iterations

    dispatch_apply(Length, queue, ^(size_t i) {

        sum += pArray[i];

    });

    

    NSLog(@" >> sum: %d", sum);

    

    dispatch_release(queue);

    

    

    /*

     注意这里使用了全局 dispatch_queue

     dispatch_apply 的定义如下:

     dispatch_apply(size_t iterations, dispatch_queue_t queue, void (^block)(size_t));

     参数 iterations 表示迭代的次数,void (^block)(size_t)  block 循环体。这么做与 for 循环相比有什么好处呢?

     答案是:并行,这里的求和是并行的,并不是按照顺序依次执行求和的。

     */

}


- (void)testBlock11

{

    /*

     我们可以将完成一组相关任务的 block 添加到一个 dispatch group 中去,这样可以在 group 中所有 block 任务都完成之后,再做其他事情。

     比如 testBlock9 中的示例也可以使用 dispatch group 实现

     */

    

    initData();

    

    __block int sum = 0;

    

    // Create a semaphore with 0 resource

    __block dispatch_semaphore_t taskSem = dispatch_semaphore_create(0);

    

    // create dispatch semaphore

    dispatch_group_t group = dispatch_group_create();

    dispatch_queue_t queue = dispatch_queue_create("StudyBlocks",NULL);

    

    dispatch_block_t task1 = ^(void) {

        int s = 0;

        for (int i = 0; i < Length; i++)

            s += data[i];

        sum = s;

        

        NSLog(@" >> after add: %d", sum);

        

        dispatch_semaphore_signal(taskSem);

    };

    

    dispatch_block_t task2 = ^(void) {

        

        dispatch_semaphore_wait(taskSem, DISPATCH_TIME_FOREVER);

        

        int s = sum;

        for (int i = 0; i < Length; i++)

            s -= data[i];

        sum = s;

        

        NSLog(@" >> after subtract: %d", sum);

    };

    

    // Fork

    dispatch_group_async(group, queue, task1);

    dispatch_group_async(group, queue, task2);

    

    // Join

    dispatch_group_wait(group, DISPATCH_TIME_FOREVER);

    

    dispatch_release(taskSem);

    dispatch_release(queue);

    dispatch_release(group);

}


- (void)testBlock12

{

    /*

     “带有自动变量值block中表现为截获自动变量

     即保存自动变量的值

     所以在执行block语法后,即使改写block中使用的自动变量的值也不会影响block执行时自动变量的值

     该源码就在block语法后改写了block中自动变量valfmt

     但执行结果是:

     val = 10

     */

    int val = 10;

    const char *fmt = "val = %d\n";

    void (^blk)(void) = ^{printf(fmt, val);};

    

    val = 2;

    fmt = "These values were changed. val = %d\n";

    

    blk();  // 执行结果是:val = 10

}


- (void)testBlock13{

    

    // 虽然我们把Block Objects 异步分派到了串行队列上,这个还是按照FIFO原则执行它的代码

    

    __block dispatch_semaphore_t sem1 = dispatch_semaphore_create(0);

    __block dispatch_semaphore_t sem2 = dispatch_semaphore_create(0);

    __block dispatch_semaphore_t sem3 = dispatch_semaphore_create(0);

    

    dispatch_queue_t firstSerialQueue = dispatch_queue_create("com.launch.GCD.serialQueue1", 0);

    

    dispatch_async(firstSerialQueue, ^{

        NSUInteger counter = 0;

        for (counter=0; counter<5; counter++) {

            NSLog(@"First interation, counter=%d", counter);

        }

        

        dispatch_semaphore_signal(sem1);

    });

    

    dispatch_async(firstSerialQueue, ^{

        NSUInteger counter = 0;

        for (counter=0; counter<5; counter++) {

            NSLog(@"Second interation, counter=%d", counter);

        }

        dispatch_semaphore_signal(sem2);

    });

    

    dispatch_async(firstSerialQueue, ^{

        NSUInteger counter = 0;

        for (counter=0; counter<5; counter++) {

            NSLog(@"Third interation, counter=%d", counter);

        }

        dispatch_semaphore_signal(sem3);

    });

    

    dispatch_semaphore_wait(sem1, DISPATCH_TIME_FOREVER);

    dispatch_semaphore_wait(sem2, DISPATCH_TIME_FOREVER);

    dispatch_semaphore_wait(sem3, DISPATCH_TIME_FOREVER);

    

    dispatch_release(firstSerialQueue);

    dispatch_release(sem1);

    dispatch_release(sem2);

    dispatch_release(sem3);

}


- (void)testBlock14{

    

    // 虽然我们把Block Objects 异步分派到了串行队列上,这个还是按照FIFO原则执行它的代码

    // 但是我们可以创建多个串行队列,不同串行队列之间是并行的

    

    __block dispatch_semaphore_t sem1 = dispatch_semaphore_create(0);

    __block dispatch_semaphore_t sem2 = dispatch_semaphore_create(0);

    __block dispatch_semaphore_t sem3 = dispatch_semaphore_create(0);

    

    dispatch_queue_t firstSerialQueue1 = dispatch_queue_create("com.launch.GCD.serialQueue1", 0);

    dispatch_queue_t firstSerialQueue2 = dispatch_queue_create("com.launch.GCD.serialQueue2", 0);

    dispatch_queue_t firstSerialQueue3 = dispatch_queue_create("com.launch.GCD.serialQueue3", 0);

    

    dispatch_async(firstSerialQueue1, ^{

        NSUInteger counter = 0;

        for (counter=0; counter<5; counter++) {

            NSLog(@"First interation, counter=%d", counter);

        }

        

        dispatch_semaphore_signal(sem1);

    });

    

    dispatch_async(firstSerialQueue2, ^{

        NSUInteger counter = 0;

        for (counter=0; counter<5; counter++) {

            NSLog(@"Second interation, counter=%d", counter);

        }

        dispatch_semaphore_signal(sem2);

    });

    

    dispatch_async(firstSerialQueue3, ^{

        NSUInteger counter = 0;

        for (counter=0; counter<5; counter++) {

            NSLog(@"Third interation, counter=%d", counter);

        }

        dispatch_semaphore_signal(sem3);

    });

    

    dispatch_semaphore_wait(sem1, DISPATCH_TIME_FOREVER);

    dispatch_semaphore_wait(sem2, DISPATCH_TIME_FOREVER);

    dispatch_semaphore_wait(sem3, DISPATCH_TIME_FOREVER);

    

    dispatch_release(firstSerialQueue1);

    dispatch_release(firstSerialQueue2);

    dispatch_release(firstSerialQueue3);

    dispatch_release(sem1);

    dispatch_release(sem2);

    dispatch_release(sem3);

}



static dispatch_once_t onceToken;

void (^executedOnlyOnce)(void) = ^{

    static NSUInteger numberOfEntries = 0;

    numberOfEntries++;

    NSLog(@"Executed %d time(s)", numberOfEntries);

};


- (void)testBlock15{

    

    // 一个任务最多只执行一次

    dispatch_queue_t concurrentQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

    

    

    dispatch_once(&onceToken, ^{

        dispatch_async(concurrentQueue, executedOnlyOnce);

    });

    

    dispatch_once(&onceToken, ^{

        dispatch_async(concurrentQueue, executedOnlyOnce);

    });

    

    

    // 稍缓线程退出

    NSUInteger count= 0;

    while (count++ < 3) {

        

        sleep(1);

    }

}


- (void)testBlock16{

    

    // 利用dispatch_once实现单例

    /*

     @interface SingletonSample : NSObject

     

     + (id)ShareInstance;

     @end

     

     @implementation SingletonSample

     + (id)ShareInstance{

     static SingletonSample *SharedInstance = nil;

     static dispatch_once_t onceToken;

     dispatch_once(&onceToken, ^{

     SharedInstance = [[SingletonSample alloc] init];

     });

     

     return SharedInstance;

     }

     @end

     */

    

    SingletonSample *obj1 = [SingletonSample ShareInstance];

    SingletonSample *obj2 = [SingletonSample ShareInstance];

    

    NSLog(@"obj1=%@, obj2=%@", obj1, obj2);

    STAssertEquals(obj1, obj2, nil);

    

}



- (void)testBlock17{

    

    // 利用GCD延时后执行任务

    

    double delayInsenconds = 2.0;

    dispatch_time_t delayInNanoSeconds = dispatch_time(DISPATCH_TIME_NOW, delayInsenconds * NSEC_PER_SEC);

    dispatch_queue_t concurrentQueue =dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

    NSLog(@"testBlock17");

    dispatch_after(delayInNanoSeconds, concurrentQueue, ^(void){

        NSLog(@"Now do work in dispatch_after");

    });

    

    // 稍缓线程退出

    NSUInteger count= 0;

    while (count++ < 5) {

        

        sleep(1);

    }

}


@end

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C语言是一种广泛使用的编程语言,它具有高效、灵活、可移植性强等特点,被广泛应用于操作系统、嵌入式系统、数据库、编译器等领域的开发。C语言的基本语法包括变量、数据类型、运算符、控制结构(如if语句、循环语句等)、函数、指针等。在编写C程序时,需要注意变量的声明和定义、指针的使用、内存的分配与释放等问题。C语言中常用的数据结构包括: 1. 数组:一种存储同类型数据的结构,可以进行索引访问和修改。 2. 链表:一种存储不同类型数据的结构,每个节点包含数据和指向下一个节点的指针。 3. 栈:一种后进先出(LIFO)的数据结构,可以通过压入(push)和弹出(pop)操作进行数据的存储和取出。 4. 队列:一种先进先出(FIFO)的数据结构,可以通过入队(enqueue)和出队(dequeue)操作进行数据的存储和取出。 5. 树:一种存储具有父子关系的数据结构,可以通过中序遍历、前序遍历和后序遍历等方式进行数据的访问和修改。 6. 图:一种存储具有节点和边关系的数据结构,可以通过广度优先搜索、深度优先搜索等方式进行数据的访问和修改。 这些数据结构在C语言中都有相应的实现方式,可以应用于各种不同的场景。C语言中的各种数据结构都有其优缺点,下面列举一些常见的数据结构的优缺点: 数组: 优点:访问和修改元素的速度非常快,适用于需要频繁读取和修改数据的场合。 缺点:数组的长度是固定的,不适合存储大小不固定的动态数据,另外数组在内存中是连续分配的,当数组较大时可能会导致内存碎片化。 链表: 优点:可以方便地插入和删除元素,适用于需要频繁插入和删除数据的场合。 缺点:访问和修改元素的速度相对较慢,因为需要遍历链表找到指定的节点。 栈: 优点:后进先出(LIFO)的特性使得栈在处理递归和括号匹配等问题时非常方便。 缺点:栈的空间有限,当数据量较大时可能会导致栈溢出。 队列: 优点:先进先出(FIFO)的特性使得
该资源内项目源码是个人的课程设计、毕业设计,代码都测试ok,都是运行成功后才上传资源,答辩评审平均分达到96分,放心下载使用! ## 项目备注 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.md文件(如有),仅供学习参考, 切勿用于商业用途。 该资源内项目源码是个人的课程设计,代码都测试ok,都是运行成功后才上传资源,答辩评审平均分达到96分,放心下载使用! ## 项目备注 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.md文件(如有),仅供学习参考, 切勿用于商业用途。
该资源内项目源码是个人的课程设计、毕业设计,代码都测试ok,都是运行成功后才上传资源,答辩评审平均分达到96分,放心下载使用! ## 项目备注 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.md文件(如有),仅供学习参考, 切勿用于商业用途。 该资源内项目源码是个人的课程设计,代码都测试ok,都是运行成功后才上传资源,答辩评审平均分达到96分,放心下载使用! ## 项目备注 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.md文件(如有),仅供学习参考, 切勿用于商业用途。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值