在使用C#开发winform应用程序时,经常会碰到对控件跨线程访问造成的异常。在winform中UI线程和工作线程是分开的,但在实际使用中经常会需要在工作线程更新UI线程中创建的控件。
方法1:禁用跨线程访问控件检测
.NET默认开启了禁止跨线程控件访问,在程序中将其置为false取消跨线程访问检测即可实现跨线程访问。
代码中添加如下代码:
Control.CheckForIllegalCrossThreadCalls = false;
备注:该方法虽然可以实现跨线程访问,但同时也取消了线程之间冲突访问的检查,因此可能会存在多个线程对同一控件进行同时访问,此时该控件的值难以预料。因此,在实际使用非线程安全,不推荐使用!
方法2:使用delegate和Invoke/BeginInvoke
Invoke是同步的,它会等待工作线程完成
BeginInvoke是异步的,它会创建另外一个线程去完成工作线程
在使用委托时分为3步(委托有C语言中函数指针的意味):
定义声明委托--->实例化委托--->调用委托
1)定义声明委托
修饰符 delegate 返回值类型 委托名 ( 参数列表 );
2)实例化委托
委托名 委托对象名 = new 委托名 ( 方法名 );
委托中的方法民对应的方法的返回值和参数列表的类型和数目必须一致。
3)调用委托
委托对象名 ( 参数列表 );
#define USE_DELEGATE
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
namespace SerialPort_CRC8_CRC16
{
public partial class Form1 : Form
{
#if USE_DELEGATE
/* 自定义委托,用于实现跨线程对控件的内容更新 */
public delegate void MyControlInvoke(Control sender, string data);
/// <summary>
/// 更新控件内容
/// </summary>
/// <param name="control"></param>
/// <param name="data"></param>
public void UpdataControl(Control control, string data)
{
//判断调用UpdataControl方法的线程和控件线程是否相同,不同则需使用Invoke方法,相同则直接操作
if (control.InvokeRequired == true)
{
MyControlInvoke myControlInvoke = new MyControlInvoke(UpdataControl);
//方法1:异步执行
this.BeginInvoke(myControlInvoke, new object[] { control, data });
//方法2:同步执行
//this.Invoke(myControlInvoke, new object[] { control, data });
}
else
{
control.Text