上位机用winform仍是wpf?

刚刚阅读1回复0
kewenda
kewenda
  • 管理员
  • 注册排名1
  • 经验值186140
  • 级别管理员
  • 主题37228
  • 回复0
楼主

一:大布景 1.鬼故事

那首诗源于昨晚的一名老友发送给我的dump文件,说它的流程呈现了回退,看了下贱程的主缓存栈,竟然又碰到了 OnUserPreferenceChanged 引致的挂死问题,实的是典范之做中的典范之做,缓存栈如下表所示:

0:000:x86> !clrstack OS Thread Id: 0x4eb688 (0) Child SP IP Call Site 002fed38 0000002b [HelperMethodFrame_1OBJ: 002fed38] System.Threading.WaitHandle.WaitOneNative(System.Runtime.InteropServices.SafeHandle, UInt32, Boolean, Boolean) 002fee1c 5cddad21 System.Threading.WaitHandle.InternalWaitOne(System.Runtime.InteropServices.SafeHandle, Int64, Boolean, Boolean) 002fee34 5cddace8 System.Threading.WaitHandle.WaitOne(Int32, Boolean) 002fee48 538d876c System.Windows.Forms.Control.WaitForWaitHandle(System.Threading.WaitHandle) 002fee88 53c5214a System.Windows.Forms.Control.MarshaledInvoke(System.Windows.Forms.Control, System.Delegate, System.Object[], Boolean) 002fee8c 538dab4b [InlinedCallFrame: 002fee8c] 002fef14 538dab4b System.Windows.Forms.Control.Invoke(System.Delegate, System.Object[]) 002fef48 53b03bc6 System.Windows.Forms.WindowsFormsSynchronizationContext.Send(System.Threading.SendOrPostCallback, System.Object) 002fef60 5c774708 Microsoft.Win32.SystemEvents+SystemEventInvokeInfo.Invoke(Boolean, System.Object[]) 002fef94 5c6616ec Microsoft.Win32.SystemEvents.RaiseEvent(Boolean, System.Object, System.Object[]) 002fefe8 5c660cd4 Microsoft.Win32.SystemEvents.OnUserPreferenceChanged(Int32, IntPtr, IntPtr) 002ff008 5c882c98 Microsoft.Win32.SystemEvents.WindowProc(IntPtr, Int32, IntPtr, IntPtr) ...

虽说,那种dump从去年看到本年,应该不出四次了,都看烦了,其构成其原因是:

未在主缓存中聚合利用者号令行,引致用 WindowsFormsSynchronizationContext.Send 跨缓存封送时,对方无法积极响应许诺从而挂死

固然晓得其原因,但有两个十分大的可惜就是在 dump 中找不到到底是哪两个号令行,只能简单的告诉老友,让其洞悉下标识符是哪里用了工做缓存成立了 利用者号令行, 有些老友按照那个信息胜利的找出,也有老友因为各类其原因没有找出,比力可惜。

为了不让那些老友的可惜继续下去,那一则做两个控造系统归纳综合,希望期望那些老友更上一层楼。

文章已经起头之前给诸位不雅寡们郭荣先上一多量WPF蔬果:

2022年物联网红极一时,寡位小厂WPF、杨瑞麟机招工需求提拔,既冲高了薪水,也带来了合作,那对C#/WPF开发人员来说是个时机。不管是WPF新手却是改行初学者,想在挖角季十强?那实得多进修点工具了!

所以当今社会就给各人整了一多量蔬果,都是完整演唱的讲义;完整源标识符等。尽数完全免费出来,需要的爸爸妈妈别忘了!

数据材料完全免费撷取C#/WPF零此根底数据材料包:

1 C#10零此根底讲义:崭新版的崭新演唱,初学者阿宝从那已经起头吧!

2 WPF号令行选集:二百多学时的控造系统、完整的教授,全站遍及世界各地的WPF号令行讲义!

3 WPF零此根底通用架构两栖做战:从零到胡尔坎写项目快速掌握虚拟化两栖做战才能!

4 WPF杨瑞麟机西门子PLC两栖做战:对标一线小厂,测验考试进阶的完整两栖做战!

数据材料完全免费自取:

因为内容过多未便呈现,需要视频讲义和配套源标识符的爸爸妈妈,可点击下方链接

也可间接点击下方卡片:点击后可主动复造威芯号,并跳转到威信。得辛苦各人自行搜刮威信号添加。内容已做打包,添加后间接发送留意查收!

二:处理计划 1. 大布景

那个问题的构成详情,我在去年的一首诗为:记一次 .NET 某新能源汽车锂电池检测流程 UI挂死阐发 中已经做过撷取,因为 dump 中找不到问题的 Control,所以也留下了一些可惜,那一则就做个弥补。

2. 问题打破点阐发

熟悉 WinForm 底层的老友应该晓得,一旦在 工做缓存 上成立了 Control 号令行,架构会主动给那个缓存装备两个 WindowsFormsSynchronizationContext 和其底层的 MarshalingControl ,那个是有源标识符支持的,各人能够找下 Control 的构造函数,简化后的源标识符如下表所示:

public class Control : Component { internal Control(bool autoInstallSyncContext) { //*** if (autoInstallSyncContext) { WindowsFormsSynchronizationContext.InstallIfNeeded(); } } } public sealed class WindowsFormsSynchronizationContext : SynchronizationContext, IDisposable { private Control controlToSendTo; private WeakReference destinationThreadRef; public WindowsFormsSynchronizationContext() { DestinationThread = Thread.CurrentThread; Application.ThreadContext threadContext = Application.ThreadContext.FromCurrent(); if (threadContext != null) { controlToSendTo = threadContext.MarshalingControl; } } internal static void InstallIfNeeded() { try { SynchronizationContext synchronizationContext = AsyncOperationManager.SynchronizationContext; if (synchronizationContext == null || synchronizationContext.GetType() == typeof(SynchronizationContext)) { AsyncOperationManager.SynchronizationContext = new WindowsFormsSynchronizationContext(); } } finally { inSyncContextInstallation = false; } } } public sealed class WindowsFormsSynchronizationContext : SynchronizationContext, IDisposable { public WindowsFormsSynchronizationContext() { DestinationThread = Thread.CurrentThread; Application.ThreadContext threadContext = Application.ThreadContext.FromCurrent(); if (threadContext != null) { controlToSendTo = threadContext.MarshalingControl; } } } internal sealed class ThreadContext { internal Control MarshalingControl { get { lock (this) { if (marshalingControl == null) { marshalingControl = new MarshalingControl(); } return marshalingControl; } } } }

那段标识符能够挖到下面两点信息。

一旦 Control 成立在工做缓存上,那那个缓存就会安拆两个 WindowsFormsSynchronizationContext 变量,好比此时就存在两个对象了。0:000:x86> !dso OS Thread Id: 0x4eb688 (0) ESP/REG Object Name 002FEC40 025a0fb0 System.Windows.Forms.WindowsFormsSynchronizationContext ... 002FEF44 0260992c System.Object[] (System.Object[]) 002FEF48 02d69164 System.Windows.Forms.WindowsFormsSynchronizationContext ...工做缓存ID 会记录在内部的 destinationThreadRef 字段中,我们试探下 02d69164 。0:000:x86> !do 02d69164 Name: System.Windows.Forms.WindowsFormsSynchronizationContext Fields: MT Field Offset Type VT Attr Value Name ... 533c2204 4002522 8 ...ows.Forms.Control 0 instance 02d69218 controlToSendTo 5cef92d0 4002523 c System.WeakReference 0 instance 02d69178 destinationThreadRef 0:000:x86> !DumpObj /d 02d69178 Name: System.WeakReference MethodTable: 5cef92d0 EEClass: 5cabf0cc Size: 12(0xc) bytes File: C:\Windows\Microsoft.Net\assembly\GAC_32\mscorlib\v4.0_4.0.0.0__b77a5c561934e089\mscorlib.dll Fields: MT Field Offset Type VT Attr Value Name 5cee2bdc 400070a 4 System.IntPtr 1 instance 111828 m_handle 0:000:x86> !do poi(111828) Name: System.Threading.Thread Fields: MT Field Offset Type VT Attr Value Name 5cee1638 40018ca 28 System.Int32 1 instance 9 m_ManagedThreadId ...

从上面的输出中能够看到,9号缓存 曾经成立了不应成立的 Control,所以找出那个 Control 就是处理问题的关键,那也是最难的。

3. 若何找出问题 Control

以我目前的手艺实力,从 dump 中确实找不到,但我能够运行时监测,打破点就是一旦那个 Control 在工做缓存中成立,底层会摆设两个 WindowsFormsSynchronizationContext 以及 MarshalingControl 对象,我们拦截他们的聚合构培养好了。

为了便利讲述,先上一段测试标识符,在 backgroundWorker1_DoWork 办法中成立两个 Button 号令行。

namespace WindowsFormsApp1 { public partial class Form1 : Form { public Form1() { InitializeComponent(); } private void backgroundWorker1_DoWork(object sender, DoWorkEventArgs e) { Button btn = new Button(); } private void Form1_Load(object sender, EventArgs e) { } private void button1_Click(object sender, EventArgs e) { backgroundWorker1.RunWorkerAsync(); } } }

接下来在 MarshalingControl 的构造函数上下两个bp断点来主动化记录,察看 new Button 的时候能否射中。

0:007> !name2ee System_Windows_Forms_ni System.Windows.Forms.Application+MarshalingControl..ctor Module: 5b9b1000 Assembly: System.Windows.Forms.dll Token: 0600554a MethodDesc: 5b9fe594 Name: System.Windows.Forms.Application+MarshalingControl..ctor() JITTED Code Address: 5bb5d1a4 0:007> bp 5bb5d1a4 "!clrstack; gc" 0:007> g OS Thread Id: 0x249c (9) Child SP IP Call Site 067ff2f0 5bb5d1a4 System.Windows.Forms.Application+MarshalingControl..ctor() 067ff2f4 5bb70224 System.Windows.Forms.Application+ThreadContext.get_MarshalingControl() 067ff324 5bb6fe5d System.Windows.Forms.WindowsFormsSynchronizationContext..ctor() 067ff338 5bb6fd4d System.Windows.Forms.WindowsFormsSynchronizationContext.InstallIfNeeded() 067ff364 5bb6e9a0 System.Windows.Forms.Control..ctor(Boolean) 067ff41c 5bbcd5cc System.Windows.Forms.ButtonBase..ctor() 067ff428 5bbcd531 System.Windows.Forms.Button..ctor() 067ff434 02342500 WindowsFormsApp1.Form1.backgroundWorker1_DoWork(System.Object, System.ComponentModel.DoWorkEventArgs) 067ff488 630ee649 System.ComponentModel.BackgroundWorker.OnDoWork(System.ComponentModel.DoWorkEventArgs) [f:\dd\NDP\fx\src\compmod\system\componentmodel\BackgroundWorker.cs @ 107] 067ff49c 630ee55d System.ComponentModel.BackgroundWorker.WorkerThreadStart(System.Object) [f:\dd\NDP\fx\src\compmod\system\componentmodel\BackgroundWorker.cs @ 245] 067ff6a0 7c69f036 [HelperMethodFrame_PROTECTOBJ: 067ff6a0] System.Runtime.Remoting.Messaging.StackBuilderSink._PrivateProcessMessage(IntPtr, System.Object[], System.Object, System.Object[] ByRef) 067ff95c 6197c82c System.Runtime.Remoting.Messaging.StackBuilderSink.AsyncProcessMessage(System.Runtime.Remoting.Messaging.IMessage, System.Runtime.Remoting.Messaging.IMessageSink) 067ff9b0 61978274 System.Runtime.Remoting.Proxies.AgileAsyncWorkerItem.DoAsyncCall() [f:\dd\ndp\clr\src\BCL\system\runtime\remoting\remotingproxy.cs @ 760] 067ff9bc 61978238 System.Runtime.Remoting.Proxies.AgileAsyncWorkerItem.ThreadPoolCallBack(System.Object) [f:\dd\ndp\clr\src\BCL\system\runtime\remoting\remotingproxy.cs @ 753] 067ff9c0 6104e7b4 System.Threading.QueueUserWorkItemCallback.WaitCallback_Context(System.Object) [f:\dd\ndp\clr\src\BCL\system\threading\threadpool.cs @ 1274] 067ff9c8 61078604 System.Threading.ExecutionContext.RunInternal(System.Threading.ExecutionContext, System.Threading.ContextCallback, System.Object, Boolean) [f:\dd\ndp\clr\src\BCL\system\threading\executioncontext.cs @ 980] 067ffa34 61078537 System.Threading.ExecutionContext.Run(System.Threading.ExecutionContext, System.Threading.ContextCallback, System.Object, Boolean) [f:\dd\ndp\clr\src\BCL\system\threading\executioncontext.cs @ 928] 067ffa48 6104f445 System.Threading.QueueUserWorkItemCallback.System.Threading.IThreadPoolWorkItem.ExecuteWorkItem() [f:\dd\ndp\clr\src\BCL\system\threading\threadpool.cs @ 1252] 067ffa5c 6104eb7d System.Threading.ThreadPoolWorkQueue.Dispatch() [f:\dd\ndp\clr\src\BCL\system\threading\threadpool.cs @ 820] 067ffaac 6104e9db System.Threading._ThreadPoolWaitCallback.PerformWaitCallback() [f:\dd\ndp\clr\src\BCL\system\threading\threadpool.cs @ 1161] 067ffccc 7c69f036 [DebuggerU2MCatchHandlerFrame: 067ffccc]

从缓存栈能够明晰的逃踪到本来是 backgroundWorker1_DoWork 下的 Button 成立的,那就是问题的根源。。。

原文链接:https://www.cnblogs.com/huangxincheng/archive/2022/11/08/16868486.html。

0
回帖 返回旅游

上位机用winform仍是wpf? 期待您的回复!

取消
载入表情清单……
载入颜色清单……
插入网络图片

取消确定

图片上传中
编辑器信息
提示信息