StaTaskScheduler dan mesej utas STA mengepam
Soalan:
Kebuntuan berlaku apabila menggunakan StaTaskScheduler dan BlockingCollection pada utas STA kerana kekurangan pengepaman mesej.
Penyelesaian:
Gunakan MsgWaitForMultipleObjectsEx
bukannya CoWaitForMultipleHandles
untuk melaksanakan konteks penyegerakan tersuai dengan keupayaan mengepam mesej.
Arahan langkah demi langkah:
SynchronizationContext.Wait
untuk menggunakan MsgWaitForMultipleObjectsEx
untuk melaksanakan pengepaman mesej dan melihat/menghantar mesej. Arahan pelaksanaan:
MWMO_INPUTAVAILABLE
dengan bendera MsgWaitForMultipleObjectsEx
. Wait
dan PeekMessage(PM_REMOVE)
dalam kaedah DispatchMessage
konteks penyegerakan tersuai. Kelebihan:
Kod sampel:
<code class="language-csharp">// 带有消息泵送的自定义同步上下文 class CustomSynchronizationContext : SynchronizationContext { protected override void OnWait(bool unused) { const uint QS_MASK = 0x0001; const uint MWMO_INPUTAVAILABLE = 0x0004; var nativeResult = MsgWaitForMultipleObjectsEx( 0, null, WAIT_INFINITE, QS_MASK, MWMO_INPUTAVAILABLE); var msg = new MSG(); while (true) { if (nativeResult == WAIT_OBJECT_0) break; if (PeekMessage(out msg, IntPtr.Zero, 0, 0, PM_REMOVE)) { TranslateMessage(ref msg); DispatchMessage(ref msg); } nativeResult = MsgWaitForMultipleObjectsEx(0, null, 0, QS_MASK, MWMO_INPUTAVAILABLE); } } } // 使用方法 using (var staThread = new ThreadWithAffinityContext(true, true)) { staThread.Run(async () => { // 在 STA 线程上安装自定义同步上下文 SynchronizationContext.SetSynchronizationContext(new CustomSynchronizationContext()); // 使用消息泵送的 STA 线程代码... }); }</code>
Atas ialah kandungan terperinci Bagaimana untuk Menyelesaikan Kebuntuan dalam Benang STA Menggunakan StaTaskScheduler dan BlockingCollection?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!