• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            隨筆 - 224  文章 - 41  trackbacks - 0
            <2010年8月>
            25262728293031
            1234567
            891011121314
            15161718192021
            22232425262728
            2930311234

            享受編程

            常用鏈接

            留言簿(11)

            隨筆分類(159)

            隨筆檔案(224)

            文章分類(2)

            文章檔案(4)

            經典c++博客

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            原文地址:http://hi.baidu.com/%D0%A1%B0%D7%C1%B3%B5%C4%D2%B6%D7%D3/blog/item/a69dc6d2c7b3190b3af3cf8f.html

            在設計中為了讓界面與邏輯分離,我的做法是使用事件,界面只要響應事件來處理界面的顯示就行了。而事件在邏輯處理中可能由不同的線程引發,這些事件的響應方法在修改界面中的控件內容時便會引發一個異常。

            這時就用到了Control.InvokeRequired 屬性 與Invoke方法。

            MSDN中說:
            獲取一個值,該值指示調用方在對控件進行方法調用時是否必須調用 Invoke 方法,因為調用方位于創建控件所在的線程以外的線程中。
            如果控件的 Handle 是在與調用線程不同的線程上創建的(說明您必須通過 Invoke 方法對控件進行調用),則為 true;否則為 false。
            Windows 窗體中的控件被綁定到特定的線程,不具備線程安全性 。因此,如果從另一個線程調用控件的方法,那么必須使用控件的一個 Invoke 方法來將調用封送到適當的線程。該屬性可用于確定是否必須調用 Invoke 方法,當不知道什么線程擁有控件時這很有用。

            下面來說下這個的用法(我的一般做法):
            首先定義一個委托,與這個事件處理函數的簽名一樣委托,當然直接使用該事件的委托也是可以的,如:

             

            private   delegate   void InvokeCallback( string msg); 


            然后就是判斷這個屬性的值來決定是否要調用Invoke函數:

             

            void m_comm_MessageEvent( string msg)
                  {
                 if (txtMessage.InvokeRequired)
                   {
                 InvokeCallbackmsgCallback =   new InvokeCallback(m_comm_MessageEvent);
                  txtMessage.Invoke(msgCallback, new   object []   { msg } );
                } 
                 else 
                   {
                 txtMessage.Text = msg;
                } 
               } 


            說明:這個函數就是事件處理函數,txtMessage是一個文本框。
            這樣就做到了窗體中控件的線程安全性。

            ------------------

            InvokeRequired 當前線程不是創建控件的線程時為true
            比如你可以自己開一個Thread,或使用Timer的事件來訪問窗體上的控件的時候,在線程中窗體的這個屬性就是True的。

            簡單的說,如果有兩個線程,Thread A和Thread B,并且有一個Control c,是在Thread A里面new的。
            那么在Thread A里面運行的任何方法調用c.InvokeRequired都會返回false。
            相反,如果在Thread B里面運行的任何方法調用c.InvokeRequired都會返回true。
            是否是UI線程與結果無關。(通常Control所在的線程是UI線程,但是可以有例外)

            也可以認為,在new Control()的時候,control用一個變量記錄下了當前線程,在調用InvokeRequired時,返回當前線程是否不等于new的時候記錄下來的那個線程。

            --------------------

            我理解:如果InvokeRequired==true表示其它線程需要訪問控件,那么調用invoke來轉給控件owner處理。

            ps:C#多線程學習
            http://www.cnblogs.com/xugang/archive/2008/04/06/1138856.html

            using System;
            using System.Collections.Generic;
            using System.ComponentModel;
            using System.Data;
            using System.Drawing;
            using System.Text;
            using System.Windows.Forms;
            using System.Threading;

            namespace WinThreadNew
            {
                
            public partial class WinThreadNew : Form
                
            {
                    
            int pro = 0;
                    Thread backWork 
            = null;
                    
            public WinThreadNew()
                    
            {
                        InitializeComponent();
                    }


                    
            private void btnStart_Click(object sender, EventArgs e)
                    
            {
                        backWork 
            = new Thread(new ThreadStart(doWork));
                        backWork.Start();
                    }


                    
            public delegate void DelegateUpdateUIPro();
                    
            private void doWork()
                    
            {
                        
            for (int i = 0; i < 100; i++ )
                        
            {
                            pro
            ++;
                            UpdateUIPro();
                            Thread.Sleep(
            100);
                        }

                        
                    }


                    
            private void UpdateUIPro()
                    
            {
                        
            if (this.InvokeRequired)
                        
            {
                            
            this.BeginInvoke(new DelegateUpdateUIPro(UpdateUIPro));
                        }
             
                        
            else
                        
            {
                            
            this.lblPro.Text = pro.ToString() + "%";
                        }

                    }


                    
            private void Form1_FormClosed(object sender, FormClosedEventArgs e)
                    
            {
                        
            this.backWork.Abort();
                        
            this.backWork.Join();
                    }

                }

            }
            posted on 2010-08-31 21:24 漂漂 閱讀(21299) 評論(0)  編輯 收藏 引用 所屬分類: c#開發
            久久人做人爽一区二区三区| 久久精品二区| 9久久9久久精品| 国产成人久久精品二区三区| 久久久青草青青国产亚洲免观| 亚洲第一永久AV网站久久精品男人的天堂AV | 综合久久久久久中文字幕亚洲国产国产综合一区首 | 伊人久久综合热线大杳蕉下载| 久久伊人精品青青草原日本| 久久久无码人妻精品无码| 国产精品无码久久综合网| 97精品依人久久久大香线蕉97| 国产三级久久久精品麻豆三级| 久久亚洲精品无码播放| 成人久久精品一区二区三区| 亚洲国产高清精品线久久| 99999久久久久久亚洲| 区久久AAA片69亚洲| 久久久精品日本一区二区三区| 久久99国产综合精品| 久久久久久精品免费免费自慰| 精品久久久久久99人妻| 国产精品天天影视久久综合网| 蜜桃麻豆WWW久久囤产精品| 99久久精品国产一区二区蜜芽| 午夜精品久久久久久毛片| 久久影院亚洲一区| 国产精品成人久久久久三级午夜电影 | 久久久久99精品成人片试看| 久久久黄色大片| 亚洲美日韩Av中文字幕无码久久久妻妇 | 久久国产免费| 久久久久久亚洲精品无码| 久久国产成人亚洲精品影院| 久久久精品午夜免费不卡| 久久久久国产精品| 国产2021久久精品| 久久无码国产| 精品熟女少妇AV免费久久| 色综合久久无码中文字幕| 久久久久女人精品毛片|