• <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
            <2025年5月>
            27282930123
            45678910
            11121314151617
            18192021222324
            25262728293031
            1234567

            享受編程

            常用鏈接

            留言簿(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#開發
            国产成人精品久久| 午夜精品久久影院蜜桃| 久久棈精品久久久久久噜噜| 久久国产欧美日韩精品免费| 亚洲午夜无码久久久久小说| 久久亚洲美女精品国产精品| 69SEX久久精品国产麻豆| 精品人妻伦一二三区久久| 91麻豆国产精品91久久久| 狠狠久久亚洲欧美专区| 精品久久久久久99人妻| 欧美熟妇另类久久久久久不卡| 狠狠色伊人久久精品综合网| 久久综合亚洲色HEZYO社区| 国产一区二区三区久久| 少妇久久久久久被弄到高潮 | 国产∨亚洲V天堂无码久久久| 青青草原1769久久免费播放| 亚洲精品高清国产一线久久| 国产精品久久久久乳精品爆| 欧美噜噜久久久XXX| 亚洲国产成人久久综合野外| 国产成人久久激情91| 久久男人Av资源网站无码软件| 亚洲午夜福利精品久久| 精品无码久久久久久久久久| 久久久一本精品99久久精品66| 亚洲精品第一综合99久久| 久久精品国产72国产精福利| 国产一区二区三区久久精品| 99久久精品国产高清一区二区 | 麻豆亚洲AV永久无码精品久久| 久久国产成人午夜AV影院| 亚洲成色999久久网站| 国产精品美女久久久久久2018| 熟妇人妻久久中文字幕| 精产国品久久一二三产区区别| 怡红院日本一道日本久久| 日韩亚洲欧美久久久www综合网 | 国产精品无码久久综合| 国产精品禁18久久久夂久|