通過Thread使用委托的方式實現按下按鈕后,結果1和結果2中顯示計算的結果值。此案例中,在Thread里面不能對控件直接操作,因為使用Thread后相當于新建了一個線程,不能直接去訪問主線程里面的控件。否則會出現下圖中的錯誤。此時需要通過委托的方式實現跨線程操作。
?使用InvokeRequired屬性判斷要操作的控件線程是否與Thread聲明的線程是同一個線程,如果不是同一個線程則使用Invoke方法對其進行操作,Invoke中使用委托的方式,Action:代表無返回值的方法,Func:代表有返回的方法。Invoke:
1)用于在創建控件的線程上執行方法,以確保線程安全地訪問 UI 控件。
2)用于調用委托(方法)的方式來確保方法在控件所屬的線程上執行。
3)當你在非 UI 線程(如后臺線程)試圖更新 UI 控件時,會拋出 Inv
alidOperationException
,使用 Invoke
可以避免這種錯誤。
control.Invoke(delegate, parameters);
control:要被調用的控件。
delegate:要在控件所屬線程上執行的委托方法。
parameters:傳遞給該方法的參數(可選),即當delegate是有參數的方法時,需要為其傳遞的參數。
如下:按下按鈕后,通過Thread創建線程,在線程中調用Invoke方法,通過Action委托(有參和無參)實現更新主界面UI。
namespace _013_Thread跨線程訪問控件
{
public partial class Thread跨線程訪問控件 : Form
{
public Thread跨線程訪問控件()
{
InitializeComponent();
}
string str = "456789";
private void btTask_Click(object sender, EventArgs e)
{
Thread thread1 = new Thread(() =>
{
if (this.lbl1.InvokeRequired==true)
{
this.lbl1.Invoke(new Action(() =>
{
lbl1.Text = "123456";
}));
}
if (this.lbl2.InvokeRequired == true)
{
this.lbl2.Invoke(new Action<string>((st) =>
{
lbl2.Text = st;
}),str);
}
});
thread1.IsBackground = true;
thread1.Start();
}
}
}
閱讀原文:原文鏈接
該文章在 2025/4/2 14:28:18 編輯過