为什么线程池在debug-build-run中以2个线程开始?

1 c# debugging multithreading asynchronous threadpool

代码 :

public class Test
{
    public static void Main()
    {
        int threadCount = ThreadPool.ThreadCount;
        Console.WriteLine($"thread count : {threadCount}");
        
        ThreadPool.SetMinThreads(1, 2);
        Console.WriteLine(ThreadPool.SetMaxThreads(1, 2));
        
        List<Task> tlist = new();
        
        for (int i = 0; i < 3; i++)
        {
            Console.WriteLine($"repeat {i}");
            
            tlist.Add(Task.Run(() =>
            {
                threadCount = ThreadPool.ThreadCount;
                Console.WriteLine($"thread count : {threadCount}");
                return FuncAsync();
            }));
            Console.WriteLine($"repeat end {i}");
        }

        Task.WaitAll(tlist.ToArray());
    }

    public static async Task FuncAsync()
    {
        Console.WriteLine("FuncAsync");
    }
}
Run Code Online (Sandbox Code Playgroud)

此代码仅在使用“发布”构建“运行”时才起作用。

调试 跑步
调试构建 工作 不工作
发布版本 工作 工作

截屏

首先,对不起我的英语能力......

我想知道当线程池的最大容量为 1 时异步方法如何工作。但是我遇到了一些意外的错误。

当我尝试使用调试构建运行时,线程池开始时已经填充了 2 个线程。 截屏

所以,没有负担得起的线程ThreadPool,那么它就不起作用。就无限停下来吧…… 截屏

但一开始就不会出现这种情况,

  1. 使用调试构建进行调试
  2. 使用发布版本进行调试
  3. 使用发布版本运行

有谁知道为什么会出现这种情况?

can*_*on7 6

我没有安装 Rider,但 VS 中也发生了同样的事情,所以我要回答这个问题。我认为 Rider 也会发生类似的情况。

在 VS 中,我们可以启动进程,然后执行 Debug -> Attach to Process... 将调试器附加到我们启动的进程。点击“Break”,然后查看“线程”窗口,我们会看到:

线程窗口

我们可以看到主线程(它卡在Task.WaitAll)、 和工作线程上)。如果我们双击工作线程并查看“调用堆栈”窗口,我们会看到:

调用堆栈窗口

您可以看到有人打电话Task.Run。值得注意的是,这不是您的代码。正在执行但不在运行时中的代码位是:

Microsoft.Extensions.DotNetDeltaApplier.dll!StartupHook.Initialize.AnonymousMethod__0() Unknown
Microsoft.Extensions.DotNetDeltaApplier.dll!StartupHook.ConnectToPipeAndSendCapabilitiesAsync()
Microsoft.Extensions.DotNetDeltaApplier.dll!Microsoft.Extensions.HotReload.ClientInitializationPayload.Write(System.IO.Stream stream)
Run Code Online (Sandbox Code Playgroud)

什么是DotNetDeltaApplier?去谷歌!从这个答案

VS 2022 将 dll 加载到您的进程中:Microsoft.VisualStudio.Debugger.Runtime.NetCoreApp.dll、Microsoft.Extensions.DotNetDeltaApplier.dll 等。如果您禁用热重载(项目属性/调试),它应该可以工作(这可能是使用 COM在应用程序之间进行通信并重新加载)

(我们可以看到它实际上使用的是命名管道而不是 COM,但这并不重要。)

因此,VS 已将额外的 DLL 加载到我们的进程中(使用启动挂钩),以支持热重载(在其中更改源代码并将这些更改修补到正在运行的可执行文件中)。该额外的 DLL 已占用并阻塞了 ThreadPool 线程,这意味着 ThreadPool 中没有剩余线程可运行您的Task.Run.

事实上,如果您直接在文件系统上找到并运行 exe(而不是从 VS 启动),它就会运行完成。