I am having hard time in understanding Wait()
, Pulse()
, PulseAll()
. Will all of them avoid deadlock? I would appreciate if you explain how to use them?
I am having hard time in understanding Wait()
, Pulse()
, PulseAll()
. Will all of them avoid deadlock? I would appreciate if you explain how to use them?
Short version:
lock(obj) {...}
is short-hand for Monitor.Enter
/ Monitor.Exit
(with exception handling etc). If nobody else has the lock, you can get it (and run your code) - otherwise your thread is blocked until the lock is aquired (by another thread releasing it).
Deadlock typically happens when either A: two threads lock things in different orders:
thread 1: lock(objA) { lock (objB) { ... } }
thread 2: lock(objB) { lock (objA) { ... } }
(here, if they each acquire the first lock, neither can ever get the second, since neither thread can exit to release their lock)
This scenario can be minimised by always locking in the same order; and you can recover (to a degree) by using Monitor.TryEnter
(instead of Monitor.Enter
/lock
) and specifying a timeout.
or B: you can block yourself with things like winforms when thread-switching while holding a lock:
lock(obj) { // on worker
this.Invoke((MethodInvoker) delegate { // switch to UI
lock(obj) { // oopsiee!
...
}
});
}
The deadlock appears obvious above, but it isn't so obvious when you have spaghetti code; possible answers: don't thread-switch while holding locks, or use BeginInvoke
so that you can at least exit the lock (letting the UI play).
Wait
/Pulse
/PulseAll
are different; they are for signalling. I use this in this answer to signal so that:
Dequeue
: if you try to dequeue data when the queue is empty, it waits for another thread to add data, which wakes up the blocked threadEnqueue
: if you try and enqueue data when the queue is full, it waits for another thread to remove data, which wakes up the blocked threadPulse
only wakes up one thread - but I'm not brainy enough to prove that the next thread is always the one I want, so I tend to use PulseAll
, and simply re-verify the conditions before continuing; as an example:
while (queue.Count >= maxSize)
{
Monitor.Wait(queue);
}
With this approach, I can safely add other meanings of Pulse
, without my existing code assuming that "I woke up, therefore there is data" - which is handy when (in the same example) I later needed to add a Close()
method.