我目前正在玩 Grand Central Dispatch 并发现了一个名为DispatchWorkItem
的课程。文档似乎有点不完整,所以我不确定以正确的方式使用它。我创建了以下代码段,并期待一些不同的东西。我希望在调用cancel
之后该项目将被取消。但由于某种原因,迭代仍在继续。我有什么想法我做错了吗?代码似乎对我来说很好。
@IBAction func testDispatchItems() {
let queue = DispatchQueue.global(attributes:.qosUserInitiated)
let item = DispatchWorkItem { [weak self] in
for i in 0...10000000 {
print(i)
self?.heavyWork()
}
}
queue.async(execute: item)
queue.after(walltime: .now() + 2) {
item.cancel()
}
}
答案 0 :(得分:47)
GCD不执行抢先取消。因此,要停止已经启动的工作项,您必须自己测试取消。在Swift中,cancel
DispatchWorkItem
。在Objective-C中,使用dispatch_block_cancel
在您创建的块上调用dispatch_block_create
。然后,您可以使用Swift中的isCancelled
进行测试以查看是否已取消(在Objective-C中称为dispatch_block_testcancel
)。
func testDispatchItems() {
let queue = DispatchQueue.global()
var item: DispatchWorkItem!
// create work item
item = DispatchWorkItem { [weak self] in
for i in 0 ... 10_000_000 {
if item.isCancelled { break }
print(i)
self?.heavyWork()
}
item = nil // resolve strong reference cycle
}
// start it
queue.async(execute: item)
// after five seconds, stop it if it hasn't already
queue.asyncAfter(deadline: .now() + 5) { [weak item] in
item?.cancel()
}
}
或者,在Objective-C中:
- (void)testDispatchItem {
dispatch_queue_t queue = dispatch_get_global_queue(QOS_CLASS_DEFAULT, 0);
static dispatch_block_t block = nil; // either static or property
__weak typeof(self) weakSelf = self;
block = dispatch_block_create(0, ^{
for (long i = 0; i < 10000000; i++) {
if (dispatch_block_testcancel(block)) { break; }
NSLog(@"%ld", i);
[weakSelf heavyWork];
}
block = nil;
});
// start it
dispatch_async(queue, block);
// after five seconds, stop it if it hasn't already
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(5 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
if (block) { dispatch_block_cancel(block); }
});
}
答案 1 :(得分:0)
没有异步API,调用“取消”方法将取消正在运行的操作。在每种情况下,“取消”方法都会执行某些操作,以便操作可以查明是否已取消操作,操作必须不时检查,然后自行停止执行更多工作。
我不知道有问题的API,但通常它会像
for i in 0...10000000 {
if (self?.cancelled)
break;
print(i)
self?.heavyWork()
}