iOS UI异步更新:dispatch_async 与 dispatch_get_global_queue 的使用方法

GCD (Grand Central Dispatch) 是Apple公司开发的一种技术,它旨在优化多核环境中的并发操作并取代传统多线程的编程模式。 在Mac OS X 10.6和IOS 4.0之后开始支持GCD。

使用GCD的一个理由就是方便。回想一下以前的多线程编程,我们会把异步调用的代码放到另外的一个函数中,并通过NSThread开启新线程来启动这段代码。 这种跳来跳去的流程对于复杂的逻辑简直就是一场灾难。更糟糕的是,调用线程时的环境对异步代码是不可见的,如果我们需要当时的临时变量的话只有两个选择: 保存到类成员变量中或者作为参数传递过去。前者会造成很多莫名奇妙的无关类成员,而后者的功能过于有限。

 

-(void) referCurrencyRate:(UIButton *)btn{

    activity.hidden=NO;

    [activity startAnimating];

 

   // may be get param

   //

     dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

      //do lost large time works

         NSString *rate=  [NetConnect GetCurrentcyRateTester:@""];

 

         dispatch_async(dispatch_get_main_queue(), ^{

             //继续UI 更新等

             UIAlertView* alert = [[UIAlertView alloc] initWithTitle:@"提示"

                                                             message:rate

                                                            delegate:nil

                                                   cancelButtonTitle:@"确定"

                                                   otherButtonTitles:nil];

             

             [alert show];

             [activity stopAnimating];

             [activity setHidesWhenStopped:YES];

        });

    });

  

   // [NSTimer scheduledTimerWithTimeInterval:5.5 target:self selector:@selector(hid) userInfo:nil repeats:NO];

}

在上面的代码中,出现了一种奇怪的格式:

^{code...}

解释一下,当一段代码被花括号包裹并在开头放置上尖号时,我们称之为块(block)。如果你学过C语言的话(实际上,block正是apple对C的一个扩展), 你可以认为这是一个增强型的函数指针。它不仅可以当做一个变量来回传递,还可以引用本身环境之外的变量(如上面代码中的parameter)。 更进一步地说,它是apple的C扩展中闭包的实现。在block里引用的对象会自动被retain,因此你也不必担心内存的问题。

另外涉及到了三个函数

void dispatch_async(
    dispatch_queue_t queue,
    dispatch_block_t block);

dispatch_queue_t dispatch_get_global_queue(
    long priority,
    unsigned long flags); 

dispatch_get_main_queue();  

dispatch_async 函数会将传入的block块放入指定的queue里运行。这个函数是异步的,这就意味着它会立即返回而不管block是否运行结束。因此,我们可以在block里运行各种耗时的操作(如网络请求) 而同时不会阻塞UI线程。 
dispatch_get_global_queue 会获取一个全局队列,我们姑且理解为系统为我们开启的一些全局线程。我们用priority指定队列的优先级,而flag作为保留字段备用(一般为0)。 
dispatch_get_main_queue 会返回主队列,也就是UI队列。它一般用于在其它队列中异步完成了一些工作后,需要在UI队列中更新界面(比如上面代码中的[self updateUIWithResult:result])的情况。

好的,知道这些特性之后,我们可以这样理解上面的代码:利用parameter变量异步地发起一个网络请求,并在请求之后更新UI线程。

这种需求很常见,诸如页面数据刷新等,基本IOSAPP这种展示的类的东西,这种用法常见。之前,我做指示器的使用都是定时器,现在都改成这种了。

定时器也可以实现这种方法,但是这不是定时器该做的事。

posted @ 2015-07-26 22:19  小毛驴  阅读(510)  评论(0编辑  收藏  举报