4

デッドロックを作成する次のコードがあります。

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))? lockatは、 どの DOES ブロックmainが原因でまだブロックされています ! スレッドがセクションParallel.Invokeに入るのに成功した方法がわかりません。lock(this)

4

1 に答える 1