假设我创建了一个 URLSessionTask
实例:
let task = URLSession.shared.dataTask(with: url) { (data, response, error) in
print (\(Thread.current))
}
// I start the task by
task.resume()
我想了解 URLSessionTask
实例是默认在主 线程上运行还是在后台 线程中运行。所以,我打印了 Thread.current
。
当我运行我的代码时,它会打印出:
<NSThread: 0x170273980>{number = 4, name = (null)}
我的问题是:
URLSessionTask
默认运行在哪个线程?主线程还是后台线程?为什么当前线程在线程
name
中显示为 null?这是否意味着它默认在后台线程中运行? (我在主线程上看到print
的 name="main")一般来说,是否有必要使用 GCD 运行
URLSessionTask
以强制它在后台线程中运行?我问这个是因为我看到一些教程没有使用 GCD 来运行URLSessionTask
,它们只使用 GCD 在主线程中运行完成处理程序。
最佳答案
简短回答:关键观察是 URLSessionTask
总是相对于您启动它的线程异步运行。除非您明确指定,否则完成处理程序和/或委托(delegate)方法将在后台线程上运行。因此,您不必在发起请求时使用 GCD,但在完成处理程序中,我们将使用 GCD 将更新 UI 或模型的任何内容分派(dispatch)到主队列。
你问:
- Which thread the
URLSessionTask
is running by default? Main thread or background thread?
那里确实有两个问题:URLSession
在内部使用哪些线程用于其自身目的,以及将运行完成处理程序和/或委托(delegate)方法的线程。
关于前一个问题,这是一个没有在任何地方记录的内部实现细节,但它似乎创建了自己的(后台)线程,带有单独的运行循环来处理请求。但这些实现细节通常并不重要:我们确信请求是异步运行的(不会阻塞当前线程)。
后一个问题,在哪个线程上调用完成处理程序和委托(delegate)方法,通常要重要得多。除非我们另外指定,否则 URLSession
会在 URLSession
为我们创建的串行操作队列上运行完成处理程序和委托(delegate)方法。这意味着它们在后台线程上运行。
此规则的唯一异常(exception)是,如果您在实例化 URLSession
时将 OperationQueue.main
指定为 queue
参数,在这种情况下显然会将主线程用于完成处理程序和委托(delegate)方法。但即使在这种情况下,请求也是异步运行的,URLSession
不会阻塞主线程。
- Why current thread shows null in thread name? Does it mean it is running in background thread by default? (I see name="main" for print on main thread)
它在串行操作队列上运行。操作队列线程使用的线程一般没有名字。但是您可以查看 OperationQueue.current?.name
以确认正在使用哪个操作队列。
- In general, is it necessary to run
URLSessionTask
with GCD in order to force it run in background thread or not? I am asking this because I saw some tutorials doesn't use GCD to runURLSessionTask
, they only use GCD to run completion handler in main thread.
这些教程建议的流程是正确的。发起请求时不必使用 GCD。它始终相对于您启动它的队列异步运行。您唯一需要做的就是将完成处理程序或委托(delegate)方法中的相关代码分派(dispatch)到适当的队列。
具体来说,由于我们通常让 URLSession
在其自己的串行队列上运行完成处理程序,因此我们必须将 UI 更新分派(dispatch)回主队列。有时会被忽视,我们通常也会将模型更新分派(dispatch)回主队列(或使用其他一些同步机制)。
关于ios - URLSessionTask 是如何运行的,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45463996/