我的节目" 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
...
非常感谢。
答案 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);
});
}