<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    blog.Toby

      BlogJava :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理 ::
      130 隨筆 :: 2 文章 :: 150 評論 :: 0 Trackbacks

    Vs 2003用多了,裝了VS 2005就有點用不習慣了,以前在2003里面不會因為在線程里面對其它線程中的控件進行操作而報錯,到了2005這個就變成一個異常了,不過這也是為了線程之間的安全性,所以只能找新的方法來實現原來的跨線程操作了。
       在講解如何進行跨線程操作前,我抄一段摘自網上關于VS2005進行這一改動的好處的說法:
        由于Windows窗體控件本質上不是線程安全的。因此如果有兩個或多個線程適度操作某一控件的狀態(set value),則可能會迫使該控件進入一種不一致的狀態。還可能出現其他與線程相關的bug,包括爭用和死鎖的情況。所以VS2005這一改動便可以增強線程安全性。
       我想大家更關心的是如何解決這個問題,如何才能操作其它線程中的控件而不引發異常,接下來我們就來探討下這個問題:

     

    第一種方法:

    這種方法我沒用過,因為大家推薦不要使用,所以我沒去實驗過,具體方法如下(摘自網上):
        設置System.Windows.Forms.Control.CheckForIllegalCrossThreadCalls  =  false;(winform.下)如果在你的程序初始化的時候設置了這個屬性,而且在你的控件中使用的都是微軟Framework類庫中的控件的話,系統就不會再拋出你上面所說的這個錯誤了。當然這只是為了將VS2003的代碼轉換到VS2005下所使用的一種常見的方法。不建議采用;

     

    第二種方法,也是我今天主要要講的就是利用delegate和invoke這個方法:

    思路:把想對另一線程中的控件實施的操作放到一個函數中,然后使用delegate代理那個函數,并且在那個函數中加入一個判斷,用InvokeRequired來判斷調用這個函數的線程是否和控件線程在同一線程中,如果是則直接執行對控件的操作,否則利用控件的Invoke或BeginInvoke方法來執行這個代理。

    在繼續講解下去之前我們先來看一下這里提到的幾個方法(如果對以下兩個東東已經了解了就可以跳過)

     

    首先是Invoke

    Invoke的中文解釋是喚醒,它有兩種參數類型我們這里只講一種即(Delegate, Object[])

    Delegate就是前面提到的那個代理,而Object[]則是用來存放Delegate所代理函數的參數

    MSDN上關于INVOKE方法有如下說明:在擁有控件的基礎窗口句柄的線程上,用指定的參數列表執行指定委托。

    用通俗的話講就是利用控件的INVOKE方法,使該控件所在的線程執行這個代理,也就是執行我們想對控件進行的操作,相當于喚醒了這個操作;

    其次是控件的InvokeRequired這個屬性(個人翻譯為’喚醒請求’):

    MSDN上關于它的解釋是獲取一個值,該值指示調用方在對控件進行方法調用時是否必須調用Invoke方法,因為調用方位于創建控件所在的線程以外的線程中。

    有通俗的話講就是返回一個值,如果與控件屬于同一個線程,則不需要進行喚醒的請求,也就是返回值為False,否則則需要進行喚醒的請求,返回為true

    總感覺MSDN上的翻譯讓人無法一看就明白,可能是自己智力不夠吧~~

     

    最后就是我們的具體程序了:
           delegate void aa(string  s);//創建一個代理

           private void  pri(string t)//這個就是我們的函數,我們把要對控件進行的操作放在這里

           {

               if  (!richTextBox1.InvokeRequired)//判斷是否需要進行喚醒的請求,如果控件與主線程在一個線程內,可以寫成if(!InvokeRequired)

               {

                   MessageBox.Show("同一線程內");

                   richTextBox1.Text =t;

               }

               else

               {

                   MessageBox.Show("不是同一個線程");

                   aa a1 =new aa(pri);

                   Invoke(a1,new object []{t});//執行喚醒操作

               }

           }

           private void  Form1_Load(object sender, System.EventArgse)

           {

               Thread  newthread = new Thread(new ThreadStart(ttread));

               newthread.Start();

           }

           void  ttread()

           {

               pri("sdfs");

           }

    執行結果先調出一個提示框顯示“不是同一個線程”,然后跳出提示框顯示“同一線程內”,然后richTextBox1中的text值為sdfs;這樣便完成了對其它線程中的控件進行操作。

    http://cnedu.blog.ytedu.cn/archives/2008/20082892324.html

    posted on 2008-07-07 14:22 渠上月 閱讀(4098) 評論(0)  編輯  收藏 所屬分類: VS 2005
    主站蜘蛛池模板: 亚洲大尺度无码无码专线一区| 婷婷久久久亚洲欧洲日产国码AV | 在线观看特色大片免费网站| 亚洲精品视频免费观看| 美美女高清毛片视频黄的一免费| 国产青草视频在线观看免费影院| 国产精品久久亚洲一区二区| 亚洲国产精品尤物YW在线观看| 少妇亚洲免费精品| 亚洲综合色视频在线观看| 国产真人无码作爱免费视频| 自拍偷自拍亚洲精品情侣| 国产色爽免费无码视频| 亚洲卡一卡2卡三卡4卡无卡三| 最好看最新的中文字幕免费| 国产婷婷综合丁香亚洲欧洲| 永久免费视频v片www| 全部一级一级毛片免费看| 亚洲午夜久久久影院伊人| 中文字幕亚洲免费无线观看日本| 亚洲AV无码一区二区三区人| 国产成人免费片在线观看| CAOPORM国产精品视频免费| 亚洲伦理一区二区| 天堂在线免费观看中文版| 一级毛片完整版免费播放一区| 国产亚洲综合成人91精品| 91免费播放人人爽人人快乐| 色欲aⅴ亚洲情无码AV蜜桃| 久久国产成人精品国产成人亚洲| 日本高清免费观看| 亚洲精品久久无码| 久久亚洲国产中v天仙www| 午夜福利不卡片在线播放免费| 国产精品亚洲一区二区三区| 亚洲产国偷V产偷V自拍色戒| 免费观看的av毛片的网站| a毛片久久免费观看| 亚洲午夜理论片在线观看| 国产亚洲一区二区三区在线观看| 男男AV纯肉无码免费播放无码|