简化自动处理InvokeRequired代码模式
在GUI事件处理中编写传统的InvokeRequired代码模式既繁琐又令人沮丧。此模式需要检查当前线程是否可以访问控件的句柄,如果不能,则在正确的线程上调用更改。
为了解决这个问题,我们改进了Lee的方法,并开发了一种简化的方法:
<code class="language-csharp">public static void InvokeIfRequired(this Control control, MethodInvoker action) { // 检查可见性,如有必要则调用 while (!control.Visible) { System.Threading.Thread.Sleep(50); } if (control.InvokeRequired) { control.Invoke(action); } else { action(); } }</code>
此方法扩展了Control类,使用方法如下:
<code class="language-csharp">richEditControl1.InvokeIfRequired(() => { // 操作控件 richEditControl1.RtfText = value; RtfHelpers.AddMissingStyles(richEditControl1); });</code>
对于需要返回值的情况,您可以使用此替代实现:
<code class="language-csharp">private static T InvokeIfRequiredReturn<T>(this Control control, Func<T> function) { if (control.InvokeRequired) { return (T)control.Invoke(function); } else { return function(); } }</code>
除了Control之外,ISynchronizeInvoke接口也可以从这个方法中受益:
<code class="language-csharp">public static void InvokeIfRequired(this ISynchronizeInvoke obj, MethodInvoker action) { if (obj.InvokeRequired) { obj.Invoke(action, null); } else { action(); } }</code>
值得注意的是,ISynchronizeInvoke需要一个对象数组作为Invoke方法的参数列表。但是,如果没有参数,您可以传递null,正如文档中所述。
我们承认,当控件最初不可见时,有时可能会遇到误报。为了解决这个问题,我们在可见性检查中加入了50毫秒的睡眠间隔。虽然这种方法通常有效,但其有效性可能取决于您的应用程序中的具体用例和时间要求。
以上是如何在GUI事件处理中简化Indokerequred代码?的详细内容。更多信息请关注PHP中文网其他相关文章!