您好,登錄后才能下訂單哦!
C#的Winform開發中,一般是不推薦使用線程去直接訪問UI控件的。
訪問 Windows 窗體控件本質上不是線程安全的。如果有兩個或多個線程操作某一控件的狀態,則可能會迫使該控件進入一種不一致的狀態。還可能出現其他與線程相關的 bug,包括爭用情況和死鎖。但是有時在程序編寫的時候寫了一些線程直接訪問UI控件的代碼,運行時也不會報錯,就以為這樣做是可以的。比如下面的代碼,定義一個線程,并且在運行是訪問進度條控件。運行代碼后發現,一切都ok,沒有任何異常拋出。
- private void button1_Click(object sender, EventArgs e)
- {
- Thread t = new Thread(() =>
- {
- try
- {
- for (int i = 1; i <= 100; i++)
- {
- progressBar1.Value = i;
- Thread.Sleep(100);
- }
- }
- catch (Exception ex)
- {
- MessageBox.Show(ex.Message);
- }
- });
- t.Start();
- }
但是在調試的時候,卻會捕獲一個異常InvalidOperationException,并提示消息:“線程間操作無效,從不是創建控件的線程訪問它。”這點困惑了很多人。其實這一切都是和Control.CheckForIllegalCrossThreadCalls有關。注意: Control.CheckForIllegalCrossThreadCalls屬性是在 .NET Framework 2.0 版中新增的。它的作用是獲取或設置一個值,該值指示是否捕獲對錯誤線程的調用。如果設為true則會捕獲對錯誤線程的調用,反之亦反。
如果一個線程并沒有創建控件A而去訪問控件A,有時候會很幸運沒什么錯誤,但是在一些復雜的情況下會導致不可以預料的錯誤。因此將 CheckForIllegalCrossThreadCalls 設置為 true 可以更容易地查找并診斷此線程活動。
CheckForIllegalCrossThreadCalls會在Control類的靜態構造方法中調用如下的語句,Debugger.IsAttached的值表示調試器是否附到進程中:
- static Control()
- {
- //...
- checkForIllegalCrossThreadCalls = Debugger.IsAttached;
- //...
- }
因此,當debug的時候,控件的checkForIllegalCrossThreadCalls是true。但是運行的時候checkForIllegalCrossThreadCalls是false。
我們可以手動的設置該值,以此控制是否需要捕獲異常。如下的代碼:
- public partial class Form1 : Form
- {
- public Form1()
- {
- InitializeComponent();
- }
- private void Form1_Load(object sender, EventArgs e)
- {
- showCurrentCheckForIllegalCrossThreadCallsValue();
- }
- private void button1_Click(object sender, EventArgs e)
- {
- Thread t = new Thread(() =>
- {
- try
- {
- for (int i = 1; i <= 100; i++)
- {
- progressBar1.Value = i;
- Thread.Sleep(100);
- }
- }
- catch (Exception ex)
- {
- MessageBox.Show(ex.Message);
- }
- });
- t.Start();
- }
- private void showCurrentCheckForIllegalCrossThreadCallsValue()
- {
- label1.Text = "ProgressBar's CheckForIllegalCrossThreadCalls Value:" + ProgressBar.CheckForIllegalCrossThreadCalls.ToString();
- }
- private void button2_Click(object sender, EventArgs e)
- {
- ProgressBar.CheckForIllegalCrossThreadCalls = true;
- showCurrentCheckForIllegalCrossThreadCallsValue();
- }
- private void button3_Click(object sender, EventArgs e)
- {
- ProgressBar.CheckForIllegalCrossThreadCalls = false;
- showCurrentCheckForIllegalCrossThreadCallsValue();
- }
- }
程序運行的時候(非調試),ProcessBar當前的CheckForIllegalCrossThreadCalls為False,手動設置為True后,再調用線程訪問UI,得到了異常。
多做幾次實驗就明白了。
最后想說明的是,要在線程中訪問UI控件,最好使用BackgroundWorker類。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。