I have a method that accepts a block and a completion block. The first block should run in the background, while the completion block should run in whatever queue the method was called.
For the latter I always used dispatch_get_current_queue()
, but it seems like it's deprecated in iOS 6 or higher. What should I use instead?
This is fundamentally the wrong approach for the API you are describing to take. If an API accepts a block and a completion block to run, the following facts need to be true:
The "block to run" should be run on an internal queue, e.g. a queue which is private to the API and hence entirely under that API's control. The only exception to this is if the API specifically declares that the block will be run on the main queue or one of the global concurrent queues.
The completion block should always be expressed as a tuple (queue, block) unless the same assumptions as for #1 hold true, e.g. the completion block will be run on a known global queue. The completion block should furthermore be dispatched async on the passed-in queue.
These are not just stylistic points, they're entirely necessary if your API is to be safe from deadlocks or other edge-case behavior that WILL otherwise hang you from the nearest tree someday. :-)
For those who still need in queue comparing, you could compare queues by their label or specifies. Check this https://stackoverflow.com/a/23220741/1531141
The pattern of "run on whatever queue the caller was on" is appealing, but ultimately not a great idea. That queue could be a low priority queue, the main queue, or some other queue with odd properties.
My favorite approach to this is to say "the completion block runs on an implementation defined queue with these properties: x, y, z", and let the block dispatch to a particular queue if the caller wants more control than that. A typical set of properties to specify would be something like "serial, non-reentrant, and async with respect to any other application-visible queue".
** EDIT **
Catfish_Man put an example in the comments below, I'm just adding it to his answer.
The other answers are great, but for the me the answer is structural. I have a method like this that's on a Singleton:
which has two dependencies, which are:
and
That way I centralize my calls to dispatch on the other thread.
Apple had deprecated
dispatch_get_current_queue()
, but left a hole in other place, so we still able to get current dispatch queue:This works for main queue at least. Note, that
underlyingQueue
property is available since iOS 8.If you need to perform the completion block in the original queue, you also may use
OperationQueue
directly, I mean without GCD.You should be careful about your use of
dispatch_get_current_queue
in the first place. From the header file:You could do either one of two things:
Keep a reference to the queue you originally posted on (if you created it via
dispatch_queue_create
), and use that from then on.Use system defined queues via
dispatch_get_global_queue
, and keep a track of which one you're using.Effectively whilst previously relying on the system to keep track of the queue you are on, you are going to have to do it yourself.