• <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年5月>
            2526272829301
            2345678
            9101112131415
            16171819202122
            23242526272829
            303112345

            享受編程

            常用鏈接

            留言簿(11)

            隨筆分類(159)

            隨筆檔案(224)

            文章分類(2)

            文章檔案(4)

            經(jīng)典c++博客

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

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

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

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

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

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

             

            private   delegate   void InvokeCallback( string msg); 


            然后就是判斷這個屬性的值來決定是否要調(diào)用Invoke函數(shù):

             

            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;
                } 
               } 


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

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

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

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

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

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

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

            ps:C#多線程學(xué)習(xí)
            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 漂漂 閱讀(21357) 評論(0)  編輯 收藏 引用 所屬分類: c#開發(fā)
            久久亚洲日韩看片无码| 久久久久久毛片免费播放| 久久se精品一区精品二区国产| 亚洲国产成人久久综合碰碰动漫3d | 少妇熟女久久综合网色欲| 欧美黑人激情性久久| 99精品久久精品一区二区| 久久影院久久香蕉国产线看观看| 久久久久国产精品人妻| 久久精品九九亚洲精品天堂| 久久成人精品| 国产精品久久久久久影院 | 国产精品99久久久精品无码| 久久久久人妻一区二区三区vr| 精品一久久香蕉国产线看播放| 久久人人爽人人爽人人AV东京热 | 亚洲伊人久久综合中文成人网| 国产欧美一区二区久久| 国产精品久久久久久五月尺| 嫩草影院久久99| 亚洲AV日韩AV天堂久久| 亚洲人成网站999久久久综合| www性久久久com| 99国产精品久久| 久久男人AV资源网站| 久久国产成人精品麻豆| 亚洲av成人无码久久精品| 少妇久久久久久被弄到高潮| 久久久久国产精品三级网| 99久久免费只有精品国产| 久久精品成人免费网站| 99精品久久精品一区二区| 97久久超碰成人精品网站| 久久久噜噜噜久久中文福利| 亚洲欧美成人综合久久久| 伊人久久综合成人网| 香蕉久久夜色精品升级完成| 色欲综合久久躁天天躁蜜桃| 婷婷伊人久久大香线蕉AV| 热re99久久精品国99热| 久久久久人妻一区精品色|