C# 多執行緒程(Multithreading)【三】UI操作

在 C# 中,當你在使用 多執行緒(Thread) 的時候,如果嘗試直接從非 UI 執行緒存取或操作 UI 控制項(

像是 Form 上的 Label、TextBox 等),通常會遇到例外,例如:

InvalidOperationException: 跨執行緒作業無效: 存取控制項 'XXX' 時所使用的執行緒與建立控制項的執行緒不同。

這是因為 Windows Forms 的 UI 控制項不是執行緒安全的。你必須使用 主執行緒(UI 執行緒) 來操作 UI 控制項。


正確做法:使用 InvokeBeginInvoke

範例:從背景執行緒更新 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。


留言

2025

05-27C# 多執行緒程(Multithreading)【五】task
05-21C# 多執行緒程(Multithreading)【四】ThreadPool
05-19C# 多執行緒程(Multithreading)【三】UI操作
05-19C# 多執行緒程(Multithreading)【二】Thread
05-12C# 多執行緒程(Multithreading)【一】
05-05WMI (Windows管理規範)與WQL(WMI 查詢語言)
04-30WndProc 視窗處理函數【三】USB裝置插入/移除偵測
04-29WndProc 視窗處理函數【二】Windows Messages (WM_*) 分類清單
04-29WndProc 視窗處理函數【一】

2024

11-27SPI Flash 操作 (Read/Write/Erase)
11-19Rotary Encoder Switch 旋轉編碼開關
11-14Command Line Interface - CLI via UART
11-14【STM32】USB HID - Volume Control
11-13【STM32】USB Custom HID
11-12【STM32】USB HID Keyboard + Mouse
11-12【STM32】USB HID Keyboard
11-12【STM32】USB HID Mouse
10-15SSD1306 128x64 OLED 【五】Wokwi Animator
09-2432F429IDISCOVERY - - LTDC [3] + FMC (SDRAM) + FatFS
09-2432F429IDISCOVERY - - LTDC [2] + FMC (SDRAM)
09-20STM32 + FatFs + SD card via SPI【三】FatFS指令操作II
09-19STM32 + FatFs + SD card via SPI【二】FatFS指令操作
09-18STM32 + FatFs + SD card via SPI【一】移植FatFS
09-0232F429IDISCOVERY - - LTDC [1]
04-17SSD1306 128x64 OLED 【四】Adafruit / GFX Library
04-17Arduino - Serial Plotter繪圖儀
04-16SSD1306 128x64 OLED 【三】
04-15SSD1306 128x64 OLED 【二】 Datasheet
04-12SSD1306 128x64 OLED 【一】I2C版本
03-20【freeRTOS】vTaskDelay 與 vTaskDelayUntil 的差異
03-19【freeRTOS】API功能列表
03-18【freeRTOS】Day1
03-08MBR和Blank project的差別
03-05刪除註冊檔registry的資料
02-27DFU over Bluetooth Low Energy
02-27nRF Util - 使用手冊
02-26nRF Command Line Tools
02-20建立BootLoader settings
02-19Secure DFU packet (ZIP) build 建立含袐鑰的Zip檔
02-19Secure DFU via BLE
02-19Secure DFU via UART
02-16nRF Util 安裝
01-16nRF52840 ic升級成nRF52840 Dongle的程式

2023

11-21[ SEGGER Embedded Studio ] 新增header files
11-21[ SEGGER Embedded Studio ] 編譯nRF52840時遇到的問題
11-07Arduino Nano ESP32 - Debugging除錯模式
11-03Git快速入手 - 使用Git GUI
10-30Git快速入手 - 使用Git Bash
10-12程式碼高亮顯示 -- google-code-prettify

2022

11-30[EZ-PD] CCG6DF CCG6SF的Host SDK遇到編譯錯誤(一)

2019

05-27[ Eagle PCB ] 合板成品
05-23#CASE_001_USB_TOOL_RL78_G12
05-22[ Eagle PCB ] 初次洗板
05-21[ Eagle PCB ] Panelize 併板
05-20[ Eagle PCB ] 建立自己的Library及元件
05-20[ Eagle PCB ] 添加library及元件
05-20[ Eagle PCB ] Introduce

2018

04-25[ TCP test Tool ] 好用的TCP Server/Client工具
01-16RZ/A1H -[0]- Renesas RZ/A1H YR0K77210S009BE BSP環境架設

2017

12-11EZ USB Suit使用JLink online debug FX3
10-20RL78 -[12]- CS+_CACX_Lab5_LowPower mode
10-16RL78 -[11]- CS+_CACX_Lab4_ADC_溫度感測
10-13RL78 -[10]- CS+_CACX_Lab4_ADC_內部參考電壓
10-13RL78 -[9]- CS+_Lab3_I2C + MPU6050
10-13RL78 -[8]- CS+_Lab2_Uart transmit
10-12RL78 -[7]- Renesas Flash Programmer 獨立燒錄軟體
10-12RL78 -[6]- CS+_雜記
10-12RL78 -[5]- CS+_tracking variables on debug mode
10-12RL78 -[4]- CS+_顯示ROM與RAM的使用size
10-12RL78 -[3]- CS+_Lab1_Led blinking
10-12RL78 -[2]- CS+專案建立
10-12RL78 -[1]- 開發環境介紹
10-06ESP-01 -[0]- 硬體設置
10-06LinkIt 7688 program Renesas RL78/G12 by 1-wire
10-06LinkIt Smart 7688 -[3]- Build the firmware from source codes
10-06LinkIt Smart 7688 -[2]- 使用UART進入bootloader / kernel console
10-06LinkIt Smart 7688 -[1]- 使用SSH連接kernel console
10-06LinkIt Smart 7688 -[0]- 初次使用
07-14LinkIt Smart 7688 -[9]- Using MRAA SPI in Python
07-13LinkIt Smart 7688 -[8]- Using MRAA UART in Python
07-12LinkIt Smart 7688 -[7]- Using MRAA I2C in Python
07-12LinkIt Smart 7688 -[6]- Using MRAA PWM in Python
07-12LinkIt Smart 7688 -[5]- Using MRAA GPIO in Python
07-10LinkIt Smart 7688 -[4]- 雜記
06-29輕乳酪蛋糕 Cotton Cheesecake
06-26VirtualBox 的 Ubuntu與Windows 共用資料夾

2015

04-29偵測USB PnP