在主线程上调用方法?
Posted
技术标签:
【中文标题】在主线程上调用方法?【英文标题】:Calling a method on the main thread? 【发布时间】:2011-08-02 03:29:07 【问题描述】:首先,我正在为 iphone 编写代码。
我需要能够在不使用performSelectorOnMainThread
的情况下调用主线程上的方法。
我不想使用performSelectorOnMainThread
的原因是当我尝试为单元测试创建模拟时它会导致问题。
[self performSelectorOnMainThread:@Selector(doSomething) withObject:nil];
问题是我的模拟知道如何调用doSomething
,但它不知道如何调用performSelectorOnMainThread
。
那么有什么解决办法吗?
【问题讨论】:
【参考方案1】:Objective-C
dispatch_async(dispatch_get_main_queue(), ^
[self doSomething];
);
斯威夫特
DispatchQueue.main.async
self.doSomething()
旧版 Swift
dispatch_async(dispatch_get_main_queue())
self.doSomething()
【讨论】:
你刚刚用 swift 3 代码让我很开心。谢谢! 最好不要直接在块中使用 self。而是使用它的弱引用。【参考方案2】:软件中有一种说法,即添加一个间接层几乎可以解决任何问题。
让 doSomething 方法是一个间接 shell,它只执行 performSelectorOnMainThread 来调用 real_doSomething 方法来执行实际的 Something 工作。或者,如果您不想更改 doSomething 方法,请让模拟测试单元调用 doSomething_redirect_shell 方法来执行类似操作。
【讨论】:
【参考方案3】:这是在 Swift 中执行此操作的更好方法:
runThisInMainThread () -> Void in
// Run your code
self.doSomething()
func runThisInMainThread(block: dispatch_block_t)
dispatch_async(dispatch_get_main_queue(), block)
它作为标准功能包含在我的仓库中,请查看:https://github.com/goktugyil/EZSwiftExtensions
【讨论】:
这无论如何都不是更好,您创建了一个除了调用另一个函数之外什么都不做的函数。顺便说一句,可以进一步简化快速语法“()-> Void in”是不需要的 它更具人类可读性/可写性。是的,自动完成添加了“() -> Void in”。有没有办法在 void>void 闭包中禁用这种自动完成行为? 您的方法名称可能会产生误导,听起来该块会立即在主线程上执行,但事实并非如此。主队列上的 dispatch_async 将代码块添加到下一个 runloop,这个重要的行为隐藏在名为 "runThisInMainThread 的方法后面 这是预期的行为,dispatch_async 将代码添加到队列的末尾。如果您希望它立即被调用,您应该改为使用 dispatch_sync 。如果您在队列中为您已经在其中的线程执行 dispatch_sync 会导致线程锁定。在您的示例中,打印顺序为“a”、“c”、“b”。 a 和 c 在 1 个运行循环中执行,因为它们在同一范围内。 b 被添加到队列的末尾,因此有时会在队列中的其他现有项目完成时调用它 @Esqarrouth - 你确定你的dispatch_async
在调用它后阻塞了代码吗?使用async
而不是sync
的全部意义在于不阻止接下来的内容。 (当然,代码的block
会阻塞主线程上的任何其他内容,因为请求代码的目的是在主线程上执行。如果你想运行后台代码,那么你会要求一个不同的队列,而不是dispatch_get_main_queue
。)【参考方案4】:
现在在 Swift 3 中:
DispatchQueue.main.async
self.doSomething()
【讨论】:
【参考方案5】:// Draw Line
func drawPath(from polyStr: String)
DispatchQueue.main.async
let path = GMSPath(fromEncodedPath: polyStr)
let polyline = GMSPolyline(path: path)
polyline.strokeWidth = 3.0
polyline.strokeColor = #colorLiteral(red: 0.05098039216, green: 0.5764705882, blue: 0.2784313725, alpha: 1)
polyline.map = self.mapVu // Google MapView
【讨论】:
以上是关于在主线程上调用方法?的主要内容,如果未能解决你的问题,请参考以下文章
在主线程上调用 tableView.reloadData 后 UITableView 不刷新