除了 "await" 之外还有哪些情况会允许同步代码被中断

What circumstances other than "await" will allow synchronous code to be interrupted

我最近在我的异步代码中遇到了一个奇怪的错误。我在 COM 控件上调用阻塞方法,它似乎允许我在阻塞时异步继续 运行。

考虑示例代码(仅用于说明目的)

public async void Main()
{
    //assume some single threaded dispatcher. eg. WpfDispatcher

    Task doAsyncTask = DoAsync();

    Console.WriteLine("Start synchronous");
    CallSomeCOMControl();
    Console.WriteLine("End synchronous");

    await doAsyncTask;
}

public async Task DoAsync()
{
    Console.WriteLine("Start async");
    await Task.Delay(1);
    Console.WriteLine("End async");
}

在正常情况下,我希望得到以下输出:

Start async
Start synchronous
End synchronous
End async

我实际上看到的是:

Start async
Start synchronous
End async
End synchronous

现在我没有 COM 控件的源代码,但我知道这是一个非常古老的 C++ 控件,没有 async/await 的概念。然而,它是一个 Active-X 控件。

我不知道 .Net RCW 的实现细节,但我假设必须继续进行某种消息泵送以允许控件工作。我还假设这种抽水允许我继续运行。

我的假设是否正确?以及我应该注意的任何其他情况我的同步代码可能会在哪里被中断?

无论是否等待,从 DoAsync 返回的任务在方法 returns 时已经开始。

而且您使用的延迟仅为 1 毫秒。

你试过更大的延迟吗?

这很正常,当呼叫越过公寓边界时,COM 会跳动。它必须抽水,不抽水很可能造成死锁。目前尚不清楚为什么必须从代码片段中跨越该边界,当您谈论 WPF 时,它 看起来 像一个 STA 线程。可能是进程外服务器,可能是您在工作线程上创建了对象。

这与 CLR 在您阻塞 STA 线程时调用 WaitOne() 或 Join() 的方式没有根本区别。这引起的重入问题与 DoEvents() 引起的痛苦非常相似。尽管 COM 和 CLR 都有选择性地泵送,但并不那么危险。尽管高度未记录。

COM 单元极大地简化了线程处理,解决了 98% 的常见问题。最后 2% 会给你带来极度难以解决的分裂性偏头痛,重新进入是该列表的首位。解决这个问题的唯一方法是不要让 COM 为组件提供线程安全的家并自己处理它。随着 code like this.