<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 渠上月 閱讀(4104) 評論(0)  編輯  收藏 所屬分類: VS 2005
    主站蜘蛛池模板: 亚洲国产综合精品中文字幕 | 亚洲一区二区三区高清不卡 | 日韩亚洲产在线观看| 先锋影音资源片午夜在线观看视频免费播放 | 亚洲av无码潮喷在线观看| 四虎国产精品成人免费久久| 国产精品免费看久久久久 | 国产乱子伦精品免费视频| 亚洲中文字幕成人在线| a一级毛片免费高清在线| 亚洲无码精品浪潮| 拍拍拍无挡视频免费观看1000| 亚洲精品乱码久久久久久自慰| 两个人看的www免费高清| 亚洲日韩激情无码一区| 成人黄网站片免费视频| 亚洲丝袜美腿视频| 在线观看H网址免费入口| 亚洲人成图片网站| 日本免费一区尤物| 国产人成网在线播放VA免费| 久久久久久亚洲精品| 4虎永免费最新永久免费地址| 久久乐国产综合亚洲精品| 国产免费无遮挡精品视频| 久久久久久噜噜精品免费直播 | 国产精品亚洲片在线| 在线观看www日本免费网站| 亚洲精品午夜国产va久久| 免费亚洲视频在线观看| 久久国产精品成人免费| 亚洲国产精品成人精品软件| 午夜小视频免费观看| aa级毛片毛片免费观看久| 亚洲成人黄色在线观看| 亚洲国产一区二区视频网站| 免费观看男人吊女人视频| 亚洲AV无码精品蜜桃| 中文字幕专区在线亚洲| 四虎在线成人免费网站| 免费国产在线精品一区|