在 C# 中,當你在使用 多執行緒(Thread) 的時候,如果嘗試直接從非 UI 執行緒存取或操作 UI 控制項(
像是 Form 上的 Label、TextBox 等),通常會遇到例外,例如:
InvalidOperationException: 跨執行緒作業無效: 存取控制項 'XXX' 時所使用的執行緒與建立控制項的執行緒不同。
這是因為 Windows Forms 的 UI 控制項不是執行緒安全的。你必須使用 主執行緒(UI 執行緒) 來操作 UI 控制項。
正確做法:使用 Invoke 或 BeginInvoke
範例:從背景執行緒更新 Label 的文字
private void UpdateLabelFromThread()
{
Thread thread = new Thread(() =>
{
string newText = "從執行緒更新 UI";
// 使用 Invoke 確保在 UI 執行緒上操作 UI
if (label1.InvokeRequired)
{
label1.Invoke(new Action(() =>
{
label1.Text = newText;
}));
}
else
{
label1.Text = newText;
}
});
thread.Start();
}
補充說明
- InvokeRequired:檢查是否需要透過 Invoke 呼叫。當前執行緒不是 UI 執行緒時會回傳 true。
- Invoke:同步執行,會等操作完成。
- BeginInvoke:非同步執行,操作不會阻塞呼叫的執行緒。
更簡單方式:使用 BackgroundWorker(適用舊專案)
BackgroundWorker worker = new BackgroundWorker();
worker.DoWork += (s, e) =>
{
// 背景執行的工作
string result = "背景工作完成";
// 回傳給 RunWorkerCompleted
e.Result = result;
};
worker.RunWorkerCompleted += (s, e) =>
{
// 在 UI 執行緒操作 UI
label1.Text = e.Result.ToString();
};
worker.RunWorkerAsync();
更現代方式(建議):使用 async/await 搭配 Task
private async void button1_Click(object sender, EventArgs e)
{
label1.Text = "工作中...";
string result = await Task.Run(() =>
{
// 模擬長時間工作
Thread.Sleep(2000);
return "工作完成";
});
label1.Text = result; // 自動回到 UI 執行緒
}
這種方式不需要 Invoke,因為 await 完成後會自動切回 UI 執行緒,非常適合 WinForms 或 WPF。
留言
張貼留言