• <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>

            cc

              C++博客 :: 首頁(yè) :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
              38 隨筆 :: 14 文章 :: 21 評(píng)論 :: 0 Trackbacks

            很多時(shí)候?qū)?/span> windows 程序都需要結(jié)合多線程,在 .net 中用如下得代碼來(lái)創(chuàng)建并啟動(dòng)一個(gè)新的線程。

            public void ThreadProc();

            Thread thread = new Thread( new ThreadStart( ThreadProc ) );

            thread.IsBackground = true;

            thread.Start();

            但是很多時(shí)候,在新的線程中,我們需要與 UI 進(jìn)行交互,在 .net 中不允許我們直接這樣做。可以參考 MSDN 中的描述:

            “Windows 窗體使用單線程單元 (STA) 模型,因?yàn)?span lang="EN-US">“Windows 窗體基于本機(jī) Win32 窗口,而 Win32 窗口從本質(zhì)上而言是單元線程。STA 模型意味著可以在任何線程上創(chuàng)建窗口,但窗口一旦創(chuàng)建后就不能切換線程,并且對(duì)它的所有函數(shù)調(diào)用都必須在其創(chuàng)建線程上發(fā)生。除了 Windows 窗體之外,.NET Framework 中的類使用自由線程模型。

            STA 模型要求需從控件的非創(chuàng)建線程調(diào)用的控件上的任何方法必須被封送到(在其上執(zhí)行)該控件的創(chuàng)建線程。基類 Control 為此目的提供了若干方法(InvokeBeginInvoke EndInvoke)。Invoke生成同步方法調(diào)用;BeginInvoke生成異步方法調(diào)用。

            Windows 窗體中的控件被綁定到特定的線程,不具備線程安全性。因此,如果從另一個(gè)線程調(diào)用控件的方法,那么必須使用控件的一個(gè) Invoke 方法來(lái)將調(diào)用封送到適當(dāng)?shù)木€程。

            正如所看到的,我們必須調(diào)用 Invoke 方法,而 BeginInvoke 可以認(rèn)為是 Invoke 的異步版本。調(diào)用方法如下:

            public delegate void OutDelegate(string text);

            public void OutText(string text)

            {

            ???? txt.AppendText(text);

            ???? txt.AppendText( "\t\n" );

            }

            OutDelegate outdelegate = new OutDelegate( OutText );

            this .BeginInvoke(outdelegate, newobject[]{text});

            如果我們需要在另外一個(gè)線程里面對(duì) UI 進(jìn)行操作,我們需要一個(gè)類似 OutText 的函數(shù),還需要一個(gè)該函數(shù)的委托 delegate ,當(dāng)然,這里展示的是自定義的, .net 中還有很多其他類型的委托,可以直接使用,不需要而外聲明。例如: MethodInvoker EventHandler ,這兩種類型委托的函數(shù)外觀是固定的, MethodInvoker void Function() 類型的委托,而 EventHandler void Function(object, EventArgs) 類型的委托,第一個(gè)不支持參數(shù),第二中的參數(shù)類型和數(shù)量都是固定的,這兩種委托可以很方便的調(diào)用,但是缺乏靈活性。請(qǐng)注意 BeginInvoke 前面的對(duì)象是 this ,也就是主線程。現(xiàn)在再介紹 Control.InvokeRequired Control 是所有控件的基類,對(duì)于這個(gè)屬性 MSDN 的描述是:

            獲取一個(gè)值,該值指示調(diào)用方在對(duì)控件進(jìn)行方法調(diào)用時(shí)是否必須調(diào)用 Invoke 方法,因?yàn)檎{(diào)用方位于創(chuàng)建控件所在的線程以外的線程中。

            該屬性可用于確定是否必須調(diào)用 Invoke 方法,當(dāng)不知道什么線程擁有控件時(shí)這很有用。

            也就是說(shuō)通過(guò)判斷 InvokeRequired 可以知道是否需要用委托來(lái)調(diào)用當(dāng)前控件的一些方法,如此可以把 OutText 函數(shù)修改一下:

            public delegate void OutDelegate(string text);

            public void OutText(string text)

            {

            ???? if( txt.InvokeRequired )

            ???? {

            ???????? OutDelegate outdelegate = new OutDelegate( OutText );

            ???????? this.BeginInvoke(outdelegate, newobject[]{text});

            ???????? return;

            ???? }

            ???? txt.AppendText(text);

            ???? txt.AppendText( "\t\n" );

            }

            注意,這里的函數(shù)沒(méi)有返回,如果有返回,需要調(diào)用 Invoke 或者 EndInvoke 來(lái)獲得返回的結(jié)果,不要因?yàn)榘b而丟失了返回值。如果調(diào)用沒(méi)有完成, Invoke EndInvoke 都將會(huì)引起阻塞。

            現(xiàn)在如果我有一個(gè)線程函數(shù)如下:

            public void ThreadProc()

            {

            ???? for(int i = 0; i < 5; i++)

            ???? {

            ???????? OutText( i.ToString() );

            ???????? Thread.Sleep(1000);

            ???? }

            }

            如果循環(huán)的次數(shù)很大,或者漏了 Thread.Sleep(1000); ,那么你的 UI 肯定會(huì)停止響應(yīng),想知道原因嗎?看看 BeginInvoke 前面的對(duì)象,沒(méi)錯(cuò),就是 this ,也就是主線程,當(dāng)你的主線程不停的調(diào)用 OutText 的時(shí)候, UI 當(dāng)然會(huì)停止響應(yīng)。

            ?

            與以前 VC 中創(chuàng)建一個(gè)新的線程需要調(diào)用 AfxBeginThread 函數(shù),該函數(shù)中第一個(gè)參數(shù)就是線程函數(shù)的地址,而第二個(gè)參數(shù)是一個(gè)類型為 LPVOID 的指針類型,這個(gè)參數(shù)將傳遞給線程函數(shù)。現(xiàn)在我們沒(méi)有辦法再使用這種方法來(lái)傳遞參數(shù)了。我們需要將傳遞給線程的參數(shù)和線程函數(shù)包裝成一個(gè)單獨(dú)的類,然后在這個(gè)類的構(gòu)造函數(shù)中初始化該線程所需的參數(shù),然后再將該實(shí)例的線程函數(shù)傳遞給 Thread 類的構(gòu)造函數(shù)。代碼大致如下:

            public class ProcClass

            {

            ???? private string procParameter = "";

            ???? public ProcClass(string parameter)

            ???? {

            ???????? procParameter = parameter;

            ???? }

            ???? public void ThreadProc()

            ???? {

            ???? }

            }

            ProcClass threadProc = new ProcClass("use thread class");

            Thread thread = new Thread( new ThreadStart( threadProc.ThreadProc ) );

            thread.IsBackground = true;

            thread.Start();

            就是這樣,需要建立一個(gè)中間類來(lái)傳遞線程所需的參數(shù)。

            那么如果我的線程又需要參數(shù),又需要和 UI 進(jìn)行交互的時(shí)候該怎么辦呢?可以修改一下代碼:

            public class ProcClass

            {

            ???? private string procParameter = "";

            ???? private Form1.OutDelegate delg = null;

            ???? public ProcClass(string parameter, Form1.OutDelegate delg)

            ???? {

            ???????? procParameter = parameter;

            ???????? this.delg = delg;

            ???? }

            ???? public void ThreadProc()

            ???? {

            ???????? delg.BeginInvoke("use ProcClass.ThreadProc()", null, null);

            ???? }

            }

            ProcClass threadProc = new ProcClass("use thread class", new OutDelegate(OutText));

            Thread thread = new Thread( new ThreadStart( threadProc.ThreadProc ) );

            thread.IsBackground = true;

            thread.Start();

            這里只是我的一些理解,如果有什么錯(cuò)誤或者不當(dāng)?shù)牡胤剑瑲g迎指出。

            posted on 2006-12-21 16:05 醒目西西 閱讀(389) 評(píng)論(0)  編輯 收藏 引用 所屬分類: 編程相關(guān)
            久久午夜福利无码1000合集| 国产精品成人99久久久久91gav | 国产精品久久久久久福利漫画| 漂亮人妻被黑人久久精品| 久久不见久久见免费视频7| 亚洲午夜久久影院| 久久久久se色偷偷亚洲精品av| 99久久久精品免费观看国产| 久久国产热这里只有精品| 人妻精品久久久久中文字幕一冢本| 国产ww久久久久久久久久| 亚洲精品午夜国产va久久| 久久免费视频观看| 天堂久久天堂AV色综合| 午夜福利91久久福利| 亚洲乱亚洲乱淫久久| 久久久无码一区二区三区| 亚洲精品午夜国产va久久| 伊人久久综合热线大杳蕉下载| 亚洲精品白浆高清久久久久久 | 久久人人爽人人爽人人片AV高清| 久久免费高清视频| 精品久久久久久成人AV| 久久人人添人人爽添人人片牛牛| 99久久国产亚洲高清观看2024| 久久久精品人妻一区二区三区四| 伊人久久大香线蕉精品不卡| 久久久久久亚洲精品无码| 一本大道加勒比久久综合| 91精品国产91久久久久福利| 精品久久人妻av中文字幕| 久久久久久亚洲Av无码精品专口| 久久人人爽人人爽人人片av麻烦| 亚洲另类欧美综合久久图片区| 久久久久久毛片免费看| 久久天天躁狠狠躁夜夜av浪潮 | 欧美日韩精品久久久免费观看| 久久人人爽人人精品视频| 人人狠狠综合久久亚洲高清| 久久无码一区二区三区少妇| 人人狠狠综合久久亚洲|