gpt4 book ai didi

c# - Parallel.Invoke 和线程调用 - 说明?

转载 作者:太空狗 更新时间:2023-10-30 01:17:22 25 4
gpt4 key购买 nike

我有这段代码会造成死锁:

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);
}
}

结果:

Before 1
(and deadlock....)

我知道锁定超出代码控制范围的实例是一种不好的做法,this。但这只是为了这个问题。

我能理解为什么这里会产生死锁。

主线程获取 main 中的 lock(test) 然后一个线程开始调用 DoWorkUsingThisLock - 它试图获取对 same 实例变量的锁定并且它被卡住了(因为 t1.Wait()main)

好的

但我看过这个答案here也会造成死锁。

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);
}
}

结果是:

Before ClassTest.DoWorkUsingThisLock 1
Before ClassTest.DoWorkUsingThisLock 2
ClassTest.DoWorkUsingThisLock 1 // <---- how ?
ClassTest.DoWorkUsingThisLock Done 1

问题:

它如何在第一次调用时获取锁 (DoWorkUsingThisLock(1))? main 处的 lock 仍然被阻塞,因为 Parallel.Invoke 确实阻塞了!我不明白线程是如何成功进入lock(this)部分的。

最佳答案

Parallel 类使用当前线程来完成一部分工作。这是一个很好的性能优化,但在线程特定状态的情况下是可以观察到的。

TPL 在很多地方都有这种“内联执行”,它以不同的方式造成了很多麻烦。许多程序不是用来处理重入的。

关于c# - Parallel.Invoke 和线程调用 - 说明?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31988089/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com