天天看點

[轉] c#中 多線程通路winform控件

我們在做winform應用的時候,大部分情況下都會碰到使用多線程控制界面上控件資訊的問題。然而我們并不能用傳統方法來解決這個問題,下面我将詳細的介紹。

傳統方法:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

<code>public</code> <code>partial</code> <code>class</code> <code>Form1 : Form</code>

<code> </code><code>{</code>

<code>    </code><code>public</code> <code>Form1()</code>

<code>    </code><code>{</code>

<code>        </code><code>InitializeComponent();</code>

<code>    </code><code>}</code>

<code>    </code><code>private</code> <code>void</code> <code>Form1_Load(</code><code>object</code> <code>sender, EventArgs e)</code>

<code>        </code><code>Thread thread = </code><code>new</code> <code>Thread(ThreadFuntion);</code>

<code>        </code><code>thread.IsBackground = </code><code>true</code><code>;</code>

<code>        </code><code>thread.Start();</code>

<code>    </code><code>private</code> <code>void</code> <code>ThreadFuntion()</code>

<code>        </code><code>while</code> <code>(</code><code>true</code><code>)</code>

<code>        </code><code>{</code>

<code>            </code><code>this</code><code>.textBox1.Text = DateTime.Now.ToString();</code>

<code>            </code><code>Thread.Sleep(1000);</code>

<code>        </code><code>}</code>

<code>}</code>

運作這段代碼,我們會看到系統抛出一個異常:Cross-thread operation not valid:Control 'textBox1' accessed from  a thread other than the thread it was created on . 這是因為.net 2.0以後加強了安全機制,不允許在winform中直接跨線程通路控件的屬性。那麼怎麼解決這個問題呢,下面提供幾種方案。

      第一種方案,我們在Form1_Load()方法中加一句代碼:

<code>private</code> <code>void</code> <code>Form1_Load(</code><code>object</code> <code>sender, EventArgs e)</code>

<code>       </code><code>Control.CheckForIllegalCrossThreadCalls = </code><code>false</code><code>;</code>

<code>       </code><code>Thread thread = </code><code>new</code> <code>Thread(ThreadFuntion);</code>

<code>       </code><code>thread.IsBackground = </code><code>true</code><code>;</code>

<code>       </code><code>thread.Start();</code>

<code> </code><code>}</code>

加入這句代碼以後發現程式可以正常運作了。這句代碼就是說在這個類中我們不檢查跨線程的調用是否合法(如果沒有 加這句話運作也沒有異常,那麼說明系統以及預設的采用了不檢查的方式)。然而,這種方法不可取。我們檢視 CheckForIllegalCrossThreadCalls 這個屬性的定義,就會發現它是一個static的,也就是說無論我們在項目的什麼地方修改了這個值,他就會在全局起作用。而且像這種跨線程通路是否存在異 常,我們通常都會去檢查。如果項目中其他人修改了這個屬性,那麼我們的方案就失敗了,我們要采取另外的方案。

下面來看第二種方案,就是使用delegate和invoke來從其他線程中控制控件資訊。網上有很多人寫了這種控制方式,然而我看了很多這種文章,表明上看來是沒有什麼問題的,但是實際上并沒有解決這個問題,首先來看網絡上的那種不完善的方式:

23

24

25

26

27

28

29

30

31

32

33

34

<code>        </code><code>private</code> <code>delegate</code> <code>void</code> <code>FlushClient();</code><code>//代理</code>

<code>        </code><code>public</code> <code>Form1()</code>

<code>            </code><code>InitializeComponent();</code>

<code>        </code><code>private</code> <code>void</code> <code>Form1_Load(</code><code>object</code> <code>sender, EventArgs e)</code>

<code>            </code><code>Thread thread = </code><code>new</code> <code>Thread(CrossThreadFlush);</code>

<code>            </code><code>thread.IsBackground=</code><code>true</code><code>;</code>

<code>            </code><code>thread.Start();</code>

<code>        </code><code>private</code> <code>void</code> <code>CrossThreadFlush()</code>

<code>            </code><code>//将代理綁定到方法</code>

<code>            </code><code>FlushClient fc = </code><code>new</code> <code>FlushClient(ThreadFuntion);</code>

<code>            </code><code>this</code><code>.BeginInvoke(fc);</code><code>//調用代理</code>

<code>        </code><code>private</code> <code>void</code> <code>ThreadFuntion()</code>

<code>            </code><code>while</code> <code>(</code><code>true</code><code>)</code>

<code>            </code><code>{</code>

<code>                </code><code>this</code><code>.textBox1.Text = DateTime.Now.ToString();</code>

<code>                </code><code>Thread.Sleep(1000);</code>

<code>            </code><code>}</code>

使用這種方式我們可以看到跨線程通路的異常沒有了。但是新問題出現了,界面沒有響應了。為什麼會出現這個問題, 我們隻是讓新開的線程無限循環重新整理,理論上應該不會對主線程産生影響的。其實不然,這種方式其實相當于把這個新開的線程“注入”到了主要制線程中,它取得 了主線程的控制。隻要這個線程不傳回,那麼主線程将永遠都無法響應。就算新開的線程中不使用無限循環,使可以傳回了。這種方式的使用多線程也失去了它本來 的意義。

現在來讓我們看看推薦的解決方案:

35

36

37

38

39

<code>            </code><code>thread.IsBackground = </code><code>true</code><code>;</code>

<code>                </code><code>//将sleep和無限循環放在等待異步的外面</code>

<code>                </code><code>ThreadFunction();</code>

<code>        </code><code>private</code> <code>void</code> <code>ThreadFunction()</code>

<code>            </code><code>if</code> <code>(</code><code>this</code><code>.textBox1.InvokeRequired)</code><code>//等待異步</code>

<code>                </code><code>FlushClient fc = </code><code>new</code> <code>FlushClient(ThreadFunction);</code>

<code>                </code><code>this</code><code>.Invoke(fc);</code><code>//通過代理調用重新整理方法</code>

<code>            </code><code>else</code>

運作上述代碼,我們可以看到問題已經被解決了,通過等待異步,我們就不會總是持有主線程的控制,這樣就可以再不發生跨線程調用異常的情況下完成多線程對winform多線程控件的控制了。

對于深山老林提出的問題,我最近找到了更優的解決方案,利用了delegate的異步調用,大家可以看看:

<code>             </code><code>FlushClient fc=</code><code>new</code> <code>FlushClient(ThreadFunction);</code>

<code>             </code><code>fc.BeginInvoke(</code><code>null</code><code>,</code><code>null</code><code>);</code>

<code>              </code><code>while</code> <code>(</code><code>true</code><code>)</code>

 這種方法也可以直接簡化為(因為delegate的異步就是開了一個異步線程):

<code>{</code>

<code>         </code><code>private</code> <code>void</code> <code>ThreadFunction()</code>

<code>&lt;br&gt;}</code>

while (true)和Thread.Sleep(1000);不要放在invoke裡面去控制就行了。

沒有整理與歸納的知識,一文不值!高度概括與梳理的知識,才是自己真正的知識與技能。 永遠不要讓自己的自由、好奇、充滿創造力的想法被現實的架構所束縛,讓創造力自由成長吧! 多花時間,關心他(她)人,正如别人所關心你的。理想的騰飛與實作,沒有别人的支援與幫助,是萬萬不能的。

    本文轉自wenglabs部落格園部落格,原文連結:http://www.cnblogs.com/arxive/p/5885069.html,如需轉載請自行聯系原作者