デッドロックを作成する次のコードがあります。
void Main()
{
ClassTest test = new ClassTest();
lock(test)
{
Task t1 = new Task(() => test.DoWorkUsingThisLock(1));
t1.Start();
t1.Wait();
}
}
public class ClassTest
{
public void DoWorkUsingThisLock(int i)
{
Console.WriteLine("Before " + i);
Console.WriteLine ("Current Thread ID is = "+Thread.CurrentThread.ManagedThreadId);
lock(this)
{
Console.WriteLine("Work " + i);
Thread.Sleep(1000);
}
Console.WriteLine("Done " + i);
}
}
結果 :
1の前
(そしてデッドロック....)
これは、コードの制御を超えてインスタンスをロックするのは悪い習慣であることはわかっていますthis
。しかし、それはこの質問のためだけです。
ここでデッドロックが発生する理由が理解できます。
メインスレッドが in を取得するlock(test)
とmain
、新しいスレッドが呼び出しを開始します - そこで同じインスタンス変数DoWorkUsingThisLock
のロックを取得しようとし、スタックします ( atのため)t1.Wait()
main
わかった
しかし、デッドロックも引き起こすこの回答をここで見ました。
void Main()
{
ClassTest test = new ClassTest();
lock(test)
{
Parallel.Invoke (
() => test.DoWorkUsingThisLock(1),
() => test.DoWorkUsingThisLock(2)
);
}
}
public class ClassTest
{
public void DoWorkUsingThisLock(int i)
{
Console.WriteLine("Before ClassTest.DoWorkUsingThisLock " + i);
lock(this)
{
Console.WriteLine("ClassTest.DoWorkUsingThisLock " + i);
Thread.Sleep(1000);
}
Console.WriteLine("ClassTest.DoWorkUsingThisLock Done " + i);
}
}
結果は次のとおりです。
ClassTest.DoWorkUsingThisLock 1
前 ClassTest.DoWorkUsingThisLock 2
ClassTest.DoWorkUsingThisLock 1 // <---- どのように ?
ClassTest.DoWorkUsingThisLock 完了 1
質問:
最初の呼び出しでロックを取得したのはなぜですか ( DoWorkUsingThisLock(1)
)? lock
atは、 どの DOES ブロックmain
が原因でまだブロックされています ! スレッドがセクションParallel.Invoke
に入るのに成功した方法がわかりません。lock(this)