`

Grand Central Dispatch 笔记

 
阅读更多

https://www.evernote.com/shard/s20/sh/54fbd735-53b6-4436-9bd4-88d46b628527/22bc221c73b55dbf1ddfb5515649c239

 

 

1. GCD 使用后不用程序去管理线程的开闭,GCD会在系统层面上去动态检测系统状态,开闭线程


2. Dispatch Queues   单行(放进去的task只会等前一个执行完了才会执行下一个) 并行(放进去的task不用等前一个执行完了,但他们开始执行的次序还是FIFO的) 2种  FIFO  把task依次放入单行queue可以实现顺序执行

3. Operation Queues 可以指定任务之间的优先级  task之间的先后依赖关系


4. __block变量是可以改变的 共享的

    dispatch_queue_t aQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0);


    for (__block int i = 0; i<10000; i++) {

        dispatch_async(aQueue, ^{

            NSLog(@"%d",i);

        });

    }

2011-07-05 17:11:38.346 ttt[41418:1803] 61

2011-07-05 17:11:38.346 ttt[41418:5f03] 1292

2011-07-05 17:11:38.348 ttt[41418:1803] 4096

2011-07-05 17:11:38.348 ttt[41418:5f03] 4954

2011-07-05 17:11:38.349 ttt[41418:1803] 5823

2011-07-05 17:11:38.349 ttt[41418:5f03] 6159

2011-07-05 17:11:38.349 ttt[41418:1803] 6575

2011-07-05 17:11:38.349 ttt[41418:5f03] 6634

2011-07-05 17:11:38.350 ttt[41418:1803] 7936

2011-07-05 17:11:38.350 ttt[41418:5f03] 8428

2011-07-05 17:11:38.351 ttt[41418:1803] 8895

2011-07-05 17:11:38.351 ttt[41418:5f03] 9364

2011-07-05 17:11:38.351 ttt[41418:1803] 9836

2011-07-05 17:11:38.351 ttt[41418:5f03] 10000

2011-07-05 17:11:38.354 ttt[41418:1803] 10000

5. 普通的外部变量的值是task(block)被创建时的值(闭包)

    dispatch_queue_t aQueue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0);


    for (int i = 0; i<1000; i++) {

        dispatch_async(aQueue, ^{

            NSLog(@"%d",i);

        });

    }

2011-07-05 17:15:37.525 ttt[41697:1803] 0

2011-07-05 17:15:37.526 ttt[41697:1803] 2

2011-07-05 17:15:37.527 ttt[41697:1803] 3

2011-07-05 17:15:37.527 ttt[41697:1803] 4

2011-07-05 17:15:37.527 ttt[41697:1803] 5

2011-07-05 17:15:37.527 ttt[41697:1803] 6

2011-07-05 17:15:37.526 ttt[41697:5f03] 1

2011-07-05 17:15:37.530 ttt[41697:5f03] 8

2011-07-05 17:15:37.530 ttt[41697:5f03] 9

2011-07-05 17:15:37.530 ttt[41697:5f03] 10

2011-07-05 17:15:37.530 ttt[41697:5f03] 11

2011-07-05 17:15:37.532 ttt[41697:6203] 13


6. queue可以有结束时执行的方法

void myFinalizerFunction(){

    NSLog(@"xxx");

}


- (void)viewDidLoad {

    [superviewDidLoad];

    

    

    dispatch_queue_t queue = dispatch_queue_create("com.example.MyQueue", NULL);

    dispatch_set_context(queue, @"xxx");

    dispatch_set_finalizer_f(queue, &myFinalizerFunction);

    for (int i = 0; i<1000; i++) {

        dispatch_async(queue, ^{

            NSLog(@"%d",i);

        });

    }

    dispatch_release(queue);


7. dispatch_sync(queue,task)  会阻塞当前线程 直到queue完成了你给的task, 但queue要完成你给的task,因为queue是FIFO的,意味着要完成之前的任务,才有机会执行你刚才给的task, 相当于当前线程等待queue里面所有任务执行完毕,  所以这句话不能在当前queue的任务代码里面调用,会造成死锁

    dispatch_queue_t queue = dispatch_queue_create("com.example.MyQueue", NULL);

    dispatch_set_context(queue, @"xxx");

    dispatch_set_finalizer_f(queue, &myFinalizerFunction);

    for (int i = 0; i<10; i++) {

        dispatch_async(queue, ^{

            NSLog(@"%d",i);

        });

    }

    NSLog(@"waiting");

    dispatch_sync(queue, ^{

        NSLog(@"wait done");

    });

    dispatch_release(queue);

2011-07-05 17:54:35.479 ttt[44203:207] waiting

2011-07-05 17:54:35.479 ttt[44203:1803] 0

2011-07-05 17:54:35.481 ttt[44203:1803] 1

2011-07-05 17:54:35.482 ttt[44203:1803] 2

2011-07-05 17:54:35.482 ttt[44203:1803] 3

2011-07-05 17:54:35.483 ttt[44203:1803] 4

2011-07-05 17:54:35.483 ttt[44203:1803] 5

2011-07-05 17:54:35.484 ttt[44203:1803] 6

2011-07-05 17:54:35.484 ttt[44203:1803] 7

2011-07-05 17:54:35.485 ttt[44203:1803] 8

2011-07-05 17:54:35.485 ttt[44203:1803] 9

2011-07-05 17:54:35.486 ttt[44203:207] wait done

2011-07-05 17:54:35.487 ttt[44203:1803] xxx



    dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

    dispatch_group_t group = dispatch_group_create();

    

        // Add a task to the group

    dispatch_group_async(group, queue, ^{

        NSLog(@"first task");

    });

    

    dispatch_queue_t otherqueue = dispatch_queue_create("com.example.MyQueue", NULL);

    for (int i = 0; i<100; i++) {

        dispatch_group_async(group, otherqueue, ^{

            NSLog(@"otherqueue task");

        });

    }


    dispatch_group_wait(group, DISPATCH_TIME_FOREVER);


    NSLog(@"end waiting");


    dispatch_release(group);

2011-07-05 19:32:31.919 ttt[50138:5f03] otherqueue task

2011-07-05 19:32:31.919 ttt[50138:1803] first task

2011-07-05 19:32:31.922 ttt[50138:5f03] otherqueue task

2011-07-05 19:32:31.923 ttt[50138:5f03] otherqueue task

...

2011-07-05 19:32:32.078 ttt[50138:5f03] otherqueue task

2011-07-05 19:32:32.079 ttt[50138:5f03] otherqueue task

2011-07-05 19:32:32.080 ttt[50138:207] end waiting


9. Although you can obtain information about the underlying thread running a task, it is better to avoid doing so

10. 当你使用的是obj-c时, block会自动retain 被它包住的任何对象,当block执行完释放,所以你不必担心你在block内使用的obj-c对象会消失
当使用的是c时(CoreFoundation types in a C function), 你必须手动retain release,
void SaveArrayAsync(CFArrayRef array) {
    CFRetain(array);

    dispatch_queue_t queue = dispatch_get_global_queue(
                                 DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);

    dispatch_async(queue, ^{
        SaveToDisk(array);
        CFRelease(array);
    }
} 

11.  block的自动retain特性有时会造成互相引用,造成彼此都不能被释放, 可以用__block来解决这个问题,因为使用__block时不会retain
This copying/retaining semantics of blocks is very useful as it means your objects won’t go away while your block is being executed on some other thread, but you can also get into trouble if you happen to be using them only on one thread and the context in a block contains a reference to some parent structure. You’ll get into a retain-loop.
// inside your controller's tableView:cellForRowAtIndexPath:
cell = [[MyCell alloc] initWithBlock:^(MyCell* cell) {
           [self adjustCell:cell];
       }]; 
So here we have a block that calls a method on the controller. Simple enough. But when that cell is created, it’s init method might copy the block to ensure the block is valid at all times. The problem is, it references self. But that self is the view controller. So you have a view controller retained by one of its descendants. This creates a retained object loop that ultimately means this controller will never get released.

 __block variables are actually not retained when the lock is copied. So you can do this

// inside your controller's tableView:cellForRowAtIndexPath:
__block me = self;
cell = [[MyCell alloc] initWithBlock:^(MyCell* cell) {
           [me adjustCell:cell];
       }];

Since the self doesn’t get retained this time, we avoid the retain cycle. While it strikes me as iffy, it does work, and it’s the only way I’ve found to deal with this situation. You can’t simply avoid copying the block if you want to call it later, because the scope will be gone by the time the function that declared the block exits. That will lead to nothing but sadness.

分享到:
评论
1 楼 hhb19900618 2011-12-06  
呵呵 怎么搞不懂怎么去使用啊?

相关推荐

    iOS史上最全的OC笔记

    笔记会阐述块的定义、类型和使用,以及如何在GCD(Grand Central Dispatch)中使用块。 7. **集合类**:OC提供了NSArray、NSMutableArray、NSDictionary、NSMutableDictionary等集合类,笔记会讲解它们的使用方法和...

    多线程学习笔记

    NSThread经常被OC程序员使用,但是随着GCD(Grand Central Dispatch)的出现,它逐渐被替代。 GCD是Apple提供的一个通用的多线程API,它的设计目标之一就是替代NSThread等传统线程技术。GCD基于C语言,并且充分地...

    IOS 学习笔记补充1

    6. **多线程编程**: GCD(Grand Central Dispatch)和OperationQueue是iOS中实现并发处理的主要工具,学习如何合理调度任务,提升应用性能。 7. **Auto Layout与Size Classes**: 自动布局用于动态构建适应不同屏幕...

    ios开发笔记

    6. **多线程**:GCD(Grand Central Dispatch)和OperationQueue等并发处理技术,用于优化应用性能。 其次,笔记中的使用技巧可能涵盖: 1. **AutoLayout与Size Class**:高级界面布局技巧,适应不同屏幕尺寸和...

    ios开发学习笔记(三)

    此外,可能会涉及多线程编程,如GCD(Grand Central Dispatch)和OperationQueue,以及内存管理,包括ARC(Automatic Reference Counting)的工作原理和避免内存泄漏的方法。 总之,iOS开发学习的第三阶段涵盖了UI...

    Objective-C学习资料(内置学习笔记,各内容源码)

    8. **GCD(Grand Central Dispatch)**:GCD是Apple提供的多线程解决方案,它简化了并发编程,使开发者可以专注于任务而不是线程管理。 现在,我们来看看“Objective-C学习笔记.docx”可能包含的内容。这个文档很...

    OC学习笔记合集

    六、GCD(Grand Central Dispatch) GCD是Apple提供的多核并行处理技术,可用于创建后台队列、同步和异步任务执行,优化程序性能。 七、KVC(Key-Value Coding)和KVO(Key-Value Observing) KVC是一种间接访问...

    Object-C笔记1_代码

    此外,理解 Blocks、GCD(Grand Central Dispatch)对于进行高效的并行编程也非常重要。最后,熟悉UIKit框架,能够帮助你构建iOS应用的用户界面。 总之,Object-C的自定义类定义和调用是学习这个语言的基础,它涵盖...

    Objective-C&UIKit自学笔记

    在深入学习时,还需要了解如何使用UIKit创建视图、控制器、事件处理等,以及更高级的主题,如内存管理策略ARC(Automatic Reference Counting)、GCD(Grand Central Dispatch)并行处理、Block语法等。

    iOS 学习OC语言部分,代码和笔记,超详细

    Block常用于异步操作,如GCD(Grand Central Dispatch)。 4. KVC(Key-Value Coding)和KVO(Key-Value Observing):KVC提供一种间接访问对象属性的方式,而KVO则允许监听并响应某个属性的变化。 三、iOS开发...

    Object-C 学习笔记

    2. **NSOperationQueue**和**GCD(Grand Central Dispatch)**:用于异步任务调度和线程管理。 3. **Core Data**:持久化数据管理框架,用于存储和检索应用程序的数据。 4. **Auto Layout**:布局引擎,根据约束自动...

    ios.zip_ios

    7. **多线程与异步编程**:GCD(Grand Central Dispatch)、OperationQueue和Promise/Future等概念,对于优化性能和提高用户体验至关重要。 8. **推送通知**:如何集成Apple Push Notification服务,为用户提供实时...

    IOS 开发笔记

    7. **多线程**:了解GCD(Grand Central Dispatch)和Operation Queues是优化应用性能的关键,它们允许开发者在后台执行任务,提高用户体验。 8. **性能优化**:通过分析和优化内存使用、减少图像大小、使用懒加载...

    iOS 作品 郭老板

    5. **多线程编程**: GCD(Grand Central Dispatch)、NSOperationQueue和 NSThread 是iOS中的多线程解决方案。笔记可能会详细阐述如何利用它们优化应用性能。 6. **动画与过渡**: UIView动画、Core Animation、...

    iOS多线程网络请求,多线程下载图片

    你也可以自定义实现,利用GCD(Grand Central Dispatch)的`dispatch_async`来开启一个新的工作线程,下载图片数据后,再通过`dispatch_sync`回到主线程更新UIImageView。 标签中的“iOS”和“iphone开发”指出了这...

    SwiftNote:Swift学习笔记

    12. **GCD(Grand Central Dispatch)**:Swift内置了对并发编程的支持,GCD可以帮助开发者有效地管理多线程。 13. **SwiftUI**:SwiftNote可能还会涉及SwiftUI,这是Apple的最新界面构建框架,使用声明式编程风格...

    IOS应用源码——iPad记事本.zip

    7. **多线程编程**:为了提高用户体验,可能会使用GCD(Grand Central Dispatch)或OperationQueue进行异步操作,例如后台数据加载和保存。 8. **Notification和Delegate**:这两个机制用于组件间的通信。记事本...

    Styf学习笔记

    它们常用于异步操作,如GCD(Grand Central Dispatch)中的任务。 7. **KVC(Key-Value Coding)与KVO(Key-Value Observing)** KVC允许间接访问对象的属性,无需直接调用setter和getter。KVO则提供了一个框架,...

    泊学付费网站Swift资料.zip

    通常,这类资源会深入浅出地介绍Swift的基础概念,如变量、常量、函数、控制流、类与结构体、枚举、协议与扩展,以及更高级的主题,如闭包、泛型、GCD(Grand Central Dispatch)和SwiftUI等。 Swift的版本更新频繁...

Global site tag (gtag.js) - Google Analytics