如何从线程更新UI

时间:2016-02-14 08:42:25

标签: ios objective-c multithreading grand-central-dispatch dispatch-async

我的节目" ThreadsNQueues" - 见下文 - 在背景线程上同时用" A"和" B"填充文本行。当一个文本行填充了10个字符时,它将被附加到输出文本缓冲区(它是一个NSMutableString),并且该缓冲区应写入UITextView(self.outView)。

问题:行self.outView.text=_output;永远不会被执行(在模拟器iOS 9.2上测试):-(对不起我对iOS上的多线程全新...

所以我的问题是:这段代码出了什么问题,如何让它变得更好(需要GCD)?

#import "ViewController.h"

@interface ViewController ()

@end

@implementation ViewController

NSMutableString *_output;
NSMutableString *_line;
NSLock* _lineLock;
dispatch_queue_t _myQueue;

- (void)viewDidLoad {
    [super viewDidLoad];
    _output=[NSMutableString string];
    _lineLock=[[NSLock alloc] init];
    _line=[NSMutableString string];
    _myQueue=dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0);

    dispatch_async(dispatch_get_main_queue(), ^{
        do {
            dispatch_async(_myQueue, ^{
                if([_lineLock tryLock]) {
                    [self append:'A'];
                    [_lineLock unlock];
                }
            });
            dispatch_async(_myQueue, ^{
                if([_lineLock tryLock]) {
                    [self append:'B'];
                    [_lineLock unlock];
                }
            });
        } while (true);
    });
}

- (void)append:(char)c {
    [_line appendFormat:@"%c",c];
    if (_line.length==10) {
        // line is complete
        [_line appendString:@"\n"];
        [_output appendString:_line];
        NSLog(@"%@",_line);
        [_line setString:@""];
        dispatch_async(dispatch_get_main_queue(), ^{
            // update ui (never called !!!)
            self.outView.text=_output;
        });
    }
}

@end

Xcode控制台日志看起来不错:

2016-02-14 09:09:38.607 ThreadsNQueues[2807:62067] BAABAAAAAB
2016-02-14 09:09:38.611 ThreadsNQueues[2807:62067] BBAAAABAAA
2016-02-14 09:09:38.614 ThreadsNQueues[2807:62047] BABBAAABBA
2016-02-14 09:09:38.617 ThreadsNQueues[2807:62044] BBAAABAAAB
2016-02-14 09:09:38.619 ThreadsNQueues[2807:62067] BABBAABBAB
2016-02-14 09:09:38.622 ThreadsNQueues[2807:62047] ABABABABAB
2016-02-14 09:09:38.623 ThreadsNQueues[2807:62047] BBABBBABAA
2016-02-14 09:09:38.624 ThreadsNQueues[2807:62047] BBBBBBABBA
2016-02-14 09:09:38.627 ThreadsNQueues[2807:62047] AABAABAABA
...

非常感谢。

2 个答案:

答案 0 :(得分:2)

您在代码开头调度到主线程,然后进入无限循环,因此主线程runloop将永远不会再运行。

如果要使用相同的结构,请调度到自定义队列而不是main,然后在内部将每个异步块调度到其中一个提供的优先级队列。

更好的解决方案可能是使用操作队列,并让每个操作在完成后将自己的新副本添加到队列中。

答案 1 :(得分:0)

使用下面提到的代码更新代码,上面的答案中已经提到了不更新UI的原因:

您在代码开始时调度到主线程,然后进入无限循环,因此主线程runloop将永远不会再运行。

运行您的线程以在后台附加字符串并在主线程上更新您的UI。

- (void)viewDidLoad {
    [super viewDidLoad];
    _output=[NSMutableString string];
    _lineLock=[[NSLock alloc] init];
    _line=[NSMutableString string];


    _myQueue=dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0);

    dispatch_async(dispatch_get_global_queue( DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^(void){
        //Background Thread

        do {
            dispatch_async(_myQueue, ^{
                if([_lineLock tryLock]) {
                    [self append:'A'];
                    [_lineLock unlock];
                }
            });
            dispatch_async(_myQueue, ^{
                if([_lineLock tryLock]) {
                    [self append:'B'];
                    [_lineLock unlock];
                }
            });
        } while (true);

    });

}