• <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
            <2008年11月>
            2627282930311
            2345678
            9101112131415
            16171819202122
            23242526272829
            30123456

            享受編程

            常用鏈接

            留言簿(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 漂漂 閱讀(21298) 評論(0)  編輯 收藏 引用 所屬分類: c#開發
            国产成人精品久久| 国产一区二区精品久久凹凸| 狠狠人妻久久久久久综合| 日本免费久久久久久久网站| 久久久久亚洲AV片无码下载蜜桃 | 精品熟女少妇AV免费久久 | 国产视频久久| 91精品国产色综合久久| 国产精品久久久久久久久软件| 97久久精品国产精品青草| 亚洲国产精品婷婷久久| 国产精品岛国久久久久| 久久国产成人亚洲精品影院| 成人综合久久精品色婷婷| 久久99久久99小草精品免视看| 亚洲精品高清国产一线久久| 久久一本综合| 亚洲国产精品无码久久久久久曰 | 久久综合五月丁香久久激情| 久久久久亚洲av无码专区导航| 久久久久久久女国产乱让韩| 999久久久免费国产精品播放| 亚洲AV日韩AV永久无码久久| 久久五月精品中文字幕| 久久久久久免费一区二区三区| 久久天天躁夜夜躁狠狠| 免费一级欧美大片久久网| 91精品国产乱码久久久久久| 久久久无码人妻精品无码| 欧美色综合久久久久久| 日韩精品无码久久一区二区三| 久久免费视频观看| 久久香蕉国产线看观看99| 久久夜色精品国产噜噜噜亚洲AV| 亚洲AV无码久久精品蜜桃| 午夜精品久久久内射近拍高清| 国产精品一区二区久久精品无码 | 日韩久久久久中文字幕人妻| 久久本道综合久久伊人| 国产视频久久| 合区精品久久久中文字幕一区|