掌握GCD以及后台永久运行的代码 (使用GCD处理后台线程和UI线程的交互)

一个例子:

在iPhone上做一个下载网页的功能,就是:在iPhone上放一个按钮,单击按钮时,显示一个转动的圆圈,表示正在进行下载,下载完成后,将内容加载到界面上的一个文本控件上。

使用GCD前:

static NSOperationQueue *queue;

// 按钮的点击事件

-(void)someClick:(id)sender{

self.indicator.hidden = NO;

[self.indicator startAnimating];

queue = [NSOperationQueue alloc]init];

NSInvocationOperation *op = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(download) object:nil];

[queue addOperation:op];

}

-(void)download{

NSURL *url = [NSURL URLWithString:@"http://www.youdao.com"];

NSError *error;

NSString *data = [NSString stringWithContentOfURL:url encoding:NSUTF8StringEncoding error:&error];

if(data !=nil){

[self performSelectorOnMainThread:@selector(download_complexted:) withObject:data waitUntilDone:NO];

}else{

NSLog(@"%@",error);

}

}

-(void)download_complexted:(NSString *)data{

[self.indicator stopAnimating];

self.indicator.hidden = YES;

self.content.text = data;

}

使用GCD以前分成了三部分来写。

使用GCD后,以上三个方法可以放在一起:

self.indicator.hidden = NO;

[self.indicator startAnimating];

dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT,0),^{

NSURL *url = [NSURL URLWithString:@"http://www.youdao.com"];

NSError *error;

NSString *data = [NSString stringWithContentOfURL:url encoding:NSUTF8StringEncoding error:&error];

if(data !=nil){

dispatch_async(diapatch_get_main_queue(),^{

[self.indicator stopAnimating];

self.indicator.hidden = YES;

self.content.text = data;

});

}else{

NSLog(@"%@",error);

}

});

首先,看到代码变短了,变清楚了。

系统提供的dispatch方法

为了方便地使用GCD,苹果提供了一些方法方便我们将block放在主线程或后台线程执行,或者延后执行。使用的例子如下所示:

 //  后台执行:
dispatch_async(dispatch_get_global_queue(0, 0), ^{
// something
}); // 主线程执行:
dispatch_async(dispatch_get_main_queue(), ^{
// something
}); // 一次性执行:
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
// code to be executed once
}); // 延迟2秒执行:
double delayInSeconds = 2.0;
dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, delayInSeconds * NSEC_PER_SEC);
dispatch_after(popTime, dispatch_get_main_queue(), ^(void){
// code to be executed on the main queue after delay
});


dispatch_queue_t也可以自己定义,如要自定义queue,可以用dispatch_queue_create方法,示例如下:
 // 自定义dispatch_queue_t
dispatch_queue_t urls_queue = dispatch_queue_create("blog.devtang.com", NULL);
dispatch_async(urls_queue, ^{
   // your code
});
dispatch_release(urls_queue);

另外,GCD还有一些高级用法,例如让后台两个线程并行执行,然后等两个线程都结束后,再汇总执行结果。这个可以用dispatch_group_t、dispatch_group_async、dispatch_group_notify来实现,示例如下:

 // 合并汇总结果
dispatch_group_t group = dispatch_group_create();
dispatch_group_async(group, dispatch_get_global_queue(0,0), ^{
// 并行执行的线程一
});
dispatch_group_async(group, dispatch_get_global_queue(0,0), ^{
// 并行执行的线程二
});
dispatch_group_notify(group, dispatch_get_global_queue(0,0), ^{
// 汇总结果
});

修改block之外的变量,只需将__block引用在外部即可。

后台的运行:

在appdelegate.h

@property (assign,nonatomic)UIBackgroundTaskIdentifier backgroundUpdateTask;

在appdelegate.m

- (void)applicationDidEnterBackground:(UIApplication *)application {
   
    
    [self beginBackgroundUpdateTask];
    
    //在这里加上你需要长久运行的代码
    
    [self endBackgroundUpdateTask];
    
}
-(void)beginBackgroundUpdateTask{

self.backgroundUpdateTask = [[UIApplication sharedApplication]beginBackgroundTaskWithExpirationHandler:^{
        [self endBackgroundUpdateTask];
    }];

}
-(void)endBackgroundUpdateTask{

[[UIApplication sharedApplication]endBackgroundTask:self.backgroundUpdateTask];
    
    self.backgroundUpdateTask = UIBackgroundTaskInvalid;

}

上一篇:命令查询职责分离模式(Command Query Responsibility Segregation,CQRS)


下一篇:开源WinForms界面开发框架Management Studio 选项卡文档 插件 Office 2007蓝色风格 后台线程