在主线程上调用方法?

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
        

    

【讨论】:

以上是关于在主线程上调用方法?的主要内容,如果未能解决你的问题,请参考以下文章

iOS - 确保在主线程上执行[重复]

在主线程上调用 tableView.reloadData 后 UITableView 不刷新

别再傻傻得认为AsyncTask只可以在主线程中创建实例和调用execute方法

向主线程调用方法的结果

调用不切换回单独的线程

应该总是在主线程上调用 becomeFirstResponder() 吗?