【问题标题】:Calling a method on the main thread?在主线程上调用方法?
【发布时间】:2011-08-02 03:29:07
【问题描述】:

首先,我正在为 iphone 编写代码。 我需要能够在不使用performSelectorOnMainThread 的情况下调用主线程上的方法。 我不想使用performSelectorOnMainThread 的原因是当我尝试为单元测试创​​建模拟时它会导致问题。

[self performSelectorOnMainThread:@Selector(doSomething) withObject:nil];

问题是我的模拟知道如何调用doSomething,但它不知道如何调用performSelectorOnMainThread

那么有什么解决办法吗?

【问题讨论】:

    标签: ios objective-c iphone multithreading methods


    【解决方案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
                }
        
            }
        

        【讨论】:

          猜你喜欢
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 2022-10-08
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          相关资源
          最近更新 更多