I am trying to implement an operation queue and I have the following scenario:
NSOperation A
NSOperation B
NSOperation C
NSOperation D
NSOperationQueue queue
I start adding A
to queue
.
During the execution of A
I need to get some data from B
and I can't continue with A
until B
returns what I need.
The same situation will occur for B
depending on C
and for C
depending on D
.
To manage this, at each NSOperation
I have this code:
NSOperation *operation; //This can be A, B, C, D or any other NSOperation
[self setQueuePriority:NSOperationQueuePriorityVeryLow]; //Set the current NSOperation with low priority
[queue addOperation: operation]; //Add the operation that I want to the queue
while(!operation.isFinished && !self.isCancelled){} //I need to wait the operation that I depend before moving on with the current operation
[self setQueuePriority:NSOperationQueuePriorityNormal]; //After the while, the other operation finished so I return my priority to normal and continue
if(self.isCancelled){ //If I get out of the while because the current operation was cancelled I also cancel the other operation.
[operation cancel];
}
My problem is that when I have something like 3 or 4 NSOperations
waiting and executing the while(!operacao.isFinished && !self.isCancelled){}
my code just freeze because the NSOperation that is important to me don't get executed, even if it have higher priority.
What I tried
Adding dependency during execution time but since my NSOperation is already running I doesn't seems to have any effect.
Instead of adding the operation to queue, I can do something
[operation start]
. It works, but canceling the current operation will also cancel the other operations that I started?I can do something like
while(!operacao.isFinished && !self.isCancelled){[NSThread sleepForTimeInterval:0.001];}
. It works, but is this the correct way? Maybe there is a better solution.
In this situation how I can guarantee that the operation that I want will run and the others will wait in background? What is the correct way to solve this?
If anyone question me why I don't add the dependency before starting my queue its because an operation will need the other only if some conditions are true. I will know if I need other operation only during execution time.
Thanks for your time.
Here's two ideas for you with contrived examples. I only used two operations but you could expand the concept to any number and/or nest them as needed.
Example 1: Using Grand Central Dispatch
GCD provides lightweight "dispatch groups", which allow you to explicitly order tasks and then wait on their completion. In this case AlphaOperation creates a group and enters it, then starts BetaOperation, whose
completionBlock
causes the group to be left. When you calldispatch_group_wait
, the current thread blocks until the number of times entering the group is equal to the number of times leaving it (a lot like retain count). Don't forget to check theisCancelled
state of the operation after any potentially long-running task.Example 2: Using a local NSOperationQueue
Since you're already with working operations, another option is creating a queue as a property of AlphaOperation, then adding BetaOperation and calling
waitUntilAllOperationsAreFinished
on the queue. This has an added benefit in that you can easily cancel the queue's operations when AlphaOperation is cancelled, simply by overriding thecancel
method.I think that you are following a wrong approach.If every operation in the queue has a priority, and they must be executed in order, why not using 4 different threads?
Take an ivar that represents the state (0: no operation is completed, 1: one operation is completed, and so on), protect it with a condition:
Initalize everything (state starts with zero), then create 4 different threads with different priorities.This is an example for the selector executed by the thread A:
So all gets executed in the order that you want.
EDIT
You can do the equivalent that I did here, but using a NSOperationQueue:
By saying that you are following the wrong approach I mean that you shouldn't use a queue with 1 as maxConcurrentOperationCount. The main queue has this value set to 1 and that's the reason of your troubles.
So basically you just need to make sure the first one finishes before beginning the next? NSOperationQueue will run in parallel unless you tell it not to. You can call setMaxConcurrentOperationCount: on your operation queue and set it to one to basically turn it into a serial queue in which only one operation will run at a time.
As you've found, you can't really do this with dependencies because that only affects when an operation starts - and if you don't know you'll need the sub operations until the main one is running, so that's no good. You can't solve this problem with a single operation queue.
However, since you're already running on an operation queue, there is no need to add the further operations to a queue. Just execute them synchronously in-place. You have to wait for them to return anyway, so why not?
One approach is to manage this from outside the operation classes ie. setup the operation dependencies between A/B/C/D correctly while creating them.
Steps: (In the method that is creating these operations)
1) Create Operation A
2) If data provided by Operation B is not available, create Operation B, and make Operation A dependent on Operation B. ie. something like
operationA.addDependency(operationB);
3). repeat step 2 for C and D. (ie. B depends on C and C depends on D, if required)
4) Add the operations to queue. The queue will execute based on the dependencies ie. D, C, B, A.
Try using
setCompletionBlock:
like this:When you set a completion block on an operation, it is executed after the main task of the operation is completed or cancelled. Thus the results of the work the operation was executing are available so that you can decide whether the next operation should be added to the queue.