青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

隨筆 - 224  文章 - 41  trackbacks - 0
<2010年6月>
303112345
6789101112
13141516171819
20212223242526
27282930123
45678910

享受編程

常用鏈接

留言簿(11)

隨筆分類(159)

隨筆檔案(224)

文章分類(2)

文章檔案(4)

經典c++博客

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

原文地址:http://www.cnblogs.com/Kevin-moon/archive/2009/01/13/1374353.html

 過年前的這段時間真是舒服,沒有了平時項目發(fā)版的緊張,剩下的就是只有在網上閑逛了,哈哈!
     今天早上閑逛的時候,在CodeProject發(fā)現(xiàn)了個不錯的文章,英文好的直接去http://www.codeproject.com/KB/threads/SynchronizationContext.aspx看吧,不好,就將就的看下我的吧,呵呵!(沒有直接翻譯,不過大概的思路相同)

理解SynchronizationContext


  SynchronizationContext 類是一個基類,可提供不帶同步的自由線程上下文。 此類實現(xiàn)的同步模型的目的是使公共語言運行庫內部的異步/同步操作能夠針對不同的異步模型采取正確的行為。此模型還簡化了托管應用程序為在不同的同步環(huán)境下正常工作而必須遵循的一些要求。同步模型的提供程序可以擴展此類并為這些方法提供自己的實現(xiàn)。(來自MSDN)
  簡而言之就是允許一個線程和另外一個線程進行通訊,SynchronizationContext在通訊中充當傳輸者的角色。另外這里有個地方需要清楚的,不是每個線程都附加SynchronizationContext這個對象,只有UI線程是一直擁有的。
  這里你可能有個問題:對于UI線程來說,是如何將SynchronizationContext這個對象附加到線程上的呢?!OK,我們先從下面的代碼開始,

 

[STAThread]
static void Main()
{
    Application.EnableVisualStyles();
    Application.SetCompatibleTextRenderingDefault(
false);

    
// let's check the context here
    var context = SynchronizationContext.Current;
    
if (context == null)
        MessageBox.Show(
"No context for this thread");
    
else
        MessageBox.Show(
"We got a context");

    
// create a form
    Form1 form = new Form1();

    
// let's check it again after creating a form
    context = SynchronizationContext.Current;

    
if (context == null)
        MessageBox.Show(
"No context for this thread");
    
else
        MessageBox.Show(
"We got a context");

    
if (context == null)
        MessageBox.Show(
"No context for this thread");

    Application.Run(
new Form1());
}

運行結果:
1、No context for this thread
2、We got a context

     
     從運行結果來看,在Form1 form = new Form1()之前,SynchronizationContext對象是為空,而當實例化Form1窗體后,SynchronizationContext對象就被附加到這個線程上了。所以可以得出答案了:當Control對象被創(chuàng)建的同時,SynchronizationContext對象也會被創(chuàng)建并附加到線程上。
     好的,我們既然已經基本了解了SynchronizationContext,接下來的事情就是使用它了!

如何使用SynchronizationContext


  應用程序有兩個線程:線程A和線程B,不過線程B比較特殊,它屬于UI線程,當這兩個線程同時運行的時候,線程A有個需求:"修改UI對象的屬性",這時候如果你是線程A,你會如何去完成需求呢?!

第一種方式:
     
     在線程A上面直接去操作UI對象,這是線程B說:"線程A,你真xx,你不知道我的特殊嘛!",然后直接拋給線程A一個異常信息,線程A得到異常后,一臉的無辜和無奈.....!

第二種方式:
  InvokeRequired?!是的,當然沒問題。(解釋下,InvokeRequired屬性是每個Control對象都具有的屬性,它會返回true和false,當是true的時候,表示它在另外一個線程上面,這是必須通過Invoke,BeginInvoke這些方法來調用更新UI對象的方法,當是false的時候,有兩種情況,1:位于當前線程上面,可以通過直接去調用修改UI對象的方法,2:位于不同的線程上,不過控件或窗體的句柄不存在。對于句柄是否存在的判斷,可以通過IsHandleCreated來獲取,如果句柄不存在,是不能調用Invoke...這些方法的,這時候你必須等待句柄的創(chuàng)建
通過InvokeRequired的實現(xiàn)方式如下:
using System;
using System.Drawing;
using System.Windows.Forms;
using System.Threading;

   
public class MyFormControl : Form
   {
      
public delegate void AddListItem(String myString);
      
public AddListItem myDelegate;
      
private Button myButton;
      
private Thread myThread;
      
private ListBox myListBox;
      
public MyFormControl()
      {
         myButton 
= new Button();
         myListBox 
= new ListBox();
         myButton.Location 
= new Point(72160);
         myButton.Size 
= new Size(15232);
         myButton.TabIndex 
= 1;
         myButton.Text 
= "Add items in list box";
         myButton.Click 
+= new EventHandler(Button_Click);
         myListBox.Location 
= new Point(4832);
         myListBox.Name 
= "myListBox";
         myListBox.Size 
= new Size(20095);
         myListBox.TabIndex 
= 2;
         ClientSize 
= new Size(292273);
         Controls.AddRange(
new Control[] {myListBox,myButton});
         Text 
= " 'Control_Invoke' example ";
         myDelegate 
= new AddListItem(AddListItemMethod);
      }
      
static void Main()
      {
         MyFormControl myForm 
= new MyFormControl();
         myForm.ShowDialog();
      }
      
public void AddListItemMethod(String myString)
      {
            myListBox.Items.Add(myString);
      }
      
private void Button_Click(object sender, EventArgs e)
      {
         myThread 
= new Thread(new ThreadStart(ThreadFunction));
         myThread.Start();
      }
      
private void ThreadFunction()
      {
         MyThreadClass myThreadClassObject  
= new MyThreadClass(this);
         myThreadClassObject.Run();
      }
   }
   
public class MyThreadClass
   {
      MyFormControl myFormControl1;
      
public MyThreadClass(MyFormControl myForm)
      {
         myFormControl1 
= myForm;
      }
      String myString;

      
public void Run()
      {
         
for (int i = 1; i <= 5; i++)
         {
            myString 
= "Step number " + i.ToString() + " executed";
            Thread.Sleep(
400);
            
// Execute the specified delegate on the thread that owns
            
// 'myFormControl1' control's underlying window handle with
            
// the specified list of arguments.
            myFormControl1.Invoke(myFormControl1.myDelegate,
                                   
new Object[] {myString});
         }
      }
   }

    不過這里存在一個有爭論的地方:這種方式必須通過調用Control的Invoke方法來實現(xiàn),這就是說調用的地方必須有一個Control的引用存在。
  看下MyThreadClass類,這個類中就存在MyFormControl的引用對象。其實如果這個類放在這里是沒有任務不妥之處的,但是如果把MyThreadClass類放在業(yè)務層,這時候問題就出現(xiàn)了,從設計角度來說,業(yè)務層是不允許和UI有任何關系,所以MyFormControl的引用對象絕對不能存在于MyThreadClass類,但是不讓它存在,更新UI控件的需求就滿足不了,這種情況下,我們如何做到一種最佳方案呢!?

第三種方式:
  本文的主角:SynchronizationContext登場了。解釋之前,先讓下面的代碼做下鋪墊,
public partial class Form1 : Form
{
    
public Form1()
    {
        InitializeComponent();
    }

    
private void mToolStripButtonThreads_Click(object sender, EventArgs e)
    {
        
// let's see the thread id
        int id = Thread.CurrentThread.ManagedThreadId;
        Trace.WriteLine(
"mToolStripButtonThreads_Click thread: " + id);

        
// grab the sync context associated to this
        
// thread (the UI thread), and save it in uiContext
        
// note that this context is set by the UI thread
        
// during Form creation (outside of your control)
        
// also note, that not every thread has a sync context attached to it.
        SynchronizationContext uiContext = SynchronizationContext.Current;

        
// create a thread and associate it to the run method
        Thread thread = new Thread(Run);

        
// start the thread, and pass it the UI context,
        
// so this thread will be able to update the UI
        
// from within the thread
        thread.Start(uiContext);
    }

    
private void Run(object state)
    {
        
// lets see the thread id
        int id = Thread.CurrentThread.ManagedThreadId;
        Trace.WriteLine(
"Run thread: " + id);

        
// grab the context from the state
        SynchronizationContext uiContext = state as SynchronizationContext;

        
for (int i = 0; i < 1000; i++)
        {
            
// normally you would do some code here
            
// to grab items from the database. or some long
            
// computation
            Thread.Sleep(10);

            
// use the ui context to execute the UpdateUI method,
            
// this insure that the UpdateUI method will run on the UI thread.

            uiContext.Post(UpdateUI, 
"line " + i.ToString());
        }
    }

    
/// <summary>
    
/// This method is executed on the main UI thread.
    
/// </summary>
    private void UpdateUI(object state)
    {
        
int id = Thread.CurrentThread.ManagedThreadId;
        Trace.WriteLine(
"UpdateUI thread:" + id);
        
string text = state as string;
        mListBox.Items.Add(text);
    }
}

運行結果:

mToolStripButtonThreads_Click thread: 10
Run thread: 
3
UpdateUI thread:
10
UpdateUI thread:
10
UpdateUI thread:
10
UpdateUI thread:
10
 (x1000 times)

    程序首先在Form1窗體的mToolStripButtonThreads_Click事件中,獲取當前的SynchronizationContext對象,然后啟動另外一個線程,并且將SynchronizationContext對象傳遞給啟動的線程,啟動的線程通過SynchronizationContext對象的Post方法來調用一個委托方法UpdateUI,因為UpdateUI是執(zhí)行在主UI線程上的,所以可以通過它來修改UI上對象的信息。
    怎么樣!不錯吧,現(xiàn)在我們可以把Control引用給拋棄了,哈哈!
    如果你去查下MSDN,會發(fā)現(xiàn)SynchronizationContext還有一個Send方法,Send和Post有什么區(qū)別?

Send VS Post,以及異常處理


首先看下異常處理的情況
private void Run(object state)
{
    
// let's see the thread id
    int id = Thread.CurrentThread.ManagedThreadId;
    Trace.WriteLine(
"Run thread: " + id);

    
// grab the context from the state
    SynchronizationContext uiContext = state as SynchronizationContext;

    
for (int i = 0; i < 1000; i++)
    {
        Trace.WriteLine(
"Loop " + i.ToString());
        
// normally you would do some code here
        
// to grab items from the database. or some long
        
// computation
        Thread.Sleep(10);

        
// use the ui context to execute the UpdateUI method, this insure that the
        
// UpdateUI method will run on the UI thread.

        
try
        {
            uiContext.Send(UpdateUI, 
"line " + i.ToString());
        }
        
catch (Exception e)
        {
            Trace.WriteLine(e.Message);
        }
    }
}

/// <summary>
/// This method is executed on the main UI thread.
/// </summary>
private void UpdateUI(object state)
{
    
throw new Exception("Boom");
}

   當你運行的時候, 你可能希望在UI線程上面去拋出,但是結果往往出忽你的意料,異常信息都在Run方法的線程上被捕獲了。這時候你可能想問:WHY?!
   解釋之前,我們先看下,Send VS Post的結果:
   Send 方法啟動一個同步請求以發(fā)送消息
   Post 方法啟動一個異步請求以發(fā)送消息。    
   哈哈,異常處理的答案迎韌而解了吧!

    今天就寫到這里吧,下一篇和大家討論下SynchronizationContext是否在所有線程中都適用...

 

posted on 2010-09-07 12:10 漂漂 閱讀(995) 評論(1)  編輯 收藏 引用 所屬分類: c#開發(fā)

FeedBack:
# re: 線程之間的通訊 UI 通訊 SynchronizationContext 使用 2010-12-28 15:03 lxf
uiContext.Send(UpdateUI, "line " + i.ToString());
參數(shù)不行吧??  回復  更多評論
  
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
              亚洲一区二区免费视频| 中文网丁香综合网| 夜夜嗨av色一区二区不卡| 极品av少妇一区二区| 国产日产亚洲精品| 国产视频欧美视频| 好吊色欧美一区二区三区视频| 国产精品理论片在线观看| 国产精品美女久久久久久免费| 国产精品xnxxcom| 国产精品网站在线观看| 国产午夜精品久久久| 伊甸园精品99久久久久久| 亚洲欧洲一区二区三区| 这里只有精品视频| 香港久久久电影| 久久久欧美精品| 欧美激情国产高清| 一区二区欧美日韩视频| 欧美一级在线视频| 欧美福利一区二区| 国产精品手机视频| 亚洲成人在线免费| 亚洲伊人色欲综合网| 久热精品视频| 亚洲欧洲在线播放| 一区二区三区视频观看| 久久久久国产一区二区三区| 欧美黄色一区| 国产在线视频欧美| 亚洲深夜激情| 欧美不卡一卡二卡免费版| 在线一区二区三区四区五区| 欧美在线播放一区二区| 欧美日韩一区二区三区免费看 | 亚洲精品视频在线| 中文在线资源观看网站视频免费不卡 | 久久精品夜色噜噜亚洲aⅴ| 欧美成人一区二区三区片免费| 亚洲精品日产精品乱码不卡| 先锋资源久久| 欧美日韩视频不卡| 黄色精品一二区| 亚洲欧美一区二区视频| 亚洲成人直播| 午夜视频一区在线观看| 欧美日韩在线播放三区四区| 在线看日韩欧美| 久久久国产91| 在线视频精品一| 欧美精品xxxxbbbb| 亚洲国产三级网| 久久综合中文色婷婷| 亚洲欧美日韩国产综合精品二区 | 亚洲欧美国产va在线影院| 欧美gay视频激情| 国户精品久久久久久久久久久不卡 | 免费视频一区| 精品二区视频| 久久精品99国产精品| 亚洲一区二区高清视频| 欧美亚韩一区| 亚洲一区三区电影在线观看| 亚洲精品免费在线观看| 美女国产一区| 亚洲电影在线看| 欧美成人中文字幕| 久久免费视频在线| 亚洲高清久久| 亚洲国产精品一区二区尤物区| 久久综合五月天婷婷伊人| 在线不卡亚洲| 最新国产の精品合集bt伙计| 欧美精品aa| 亚洲一区999| 午夜精品久久久久久99热软件| 国产日本欧美一区二区三区| 久久久www| 久久精品在线播放| 亚洲国产精品激情在线观看| 亚洲国产精品激情在线观看| 欧美激情中文字幕乱码免费| 国产精品99久久99久久久二8| 亚洲欧美影音先锋| 午夜精品福利视频| 一区二区在线不卡| 亚洲欧洲久久| 国产精品成人观看视频国产奇米| 亚洲欧美变态国产另类| 性刺激综合网| 亚洲精品在线观| 亚洲一二三区精品| 雨宫琴音一区二区在线| 亚洲国产经典视频| 国产精品久久久久国产a级| 久久久天天操| 欧美日韩国产综合视频在线观看中文| 亚洲女ⅴideoshd黑人| 久久精品免费观看| 亚洲视频欧美视频| 久久久久久一区二区三区| 中国成人在线视频| 久久精品女人| 亚洲欧洲av一区二区三区久久| 久久久久久夜| 小处雏高清一区二区三区| 久久综合激情| 午夜精品久久久久久久久| 美女国内精品自产拍在线播放| 亚洲综合清纯丝袜自拍| 六月婷婷久久| 欧美亚洲一级| 欧美日韩妖精视频| 欧美成人精品福利| 国产精品人人做人人爽人人添| 欧美激情区在线播放| 国产区在线观看成人精品| 日韩视频一区二区三区在线播放| 激情久久五月天| 亚洲伊人久久综合| 这里只有精品视频在线| 免费的成人av| 另类av一区二区| 国产婷婷色一区二区三区在线| 亚洲精品一区二区三区四区高清| 激情文学综合丁香| 午夜国产一区| 欧美在线亚洲| 国产精品一区二区久久久久| 亚洲毛片在线观看.| 亚洲日韩第九十九页| 久久这里只有精品视频首页| 久久久久久久综合日本| 国产精品美女视频网站| 99视频在线观看一区三区| 亚洲美女诱惑| 欧美激情精品久久久| 欧美国产视频日韩| 亚洲国产成人久久综合一区| 久久网站免费| 欧美黄色影院| 亚洲精品久久久久久久久久久久| 久久亚洲影音av资源网| 欧美chengren| 亚洲精品一区在线观看香蕉| 欧美α欧美αv大片| 亚洲国产欧美日韩另类综合| 亚洲日本中文字幕| 欧美日韩一区二区三区视频 | 亚洲国产视频直播| 在线视频国内自拍亚洲视频| 久久一区中文字幕| 亚洲第一精品在线| 亚洲精品少妇30p| 欧美日韩国产限制| 一区二区三区国产精华| 亚洲免费小视频| 国产精品一区二区三区成人| 老司机免费视频一区二区三区| 欧美一区二区视频在线| 久久免费高清视频| 亚洲成色www久久网站| 欧美77777| 亚洲免费精品| 欧美亚洲在线| 在线色欧美三级视频| 欧美福利电影网| 一区二区三区四区五区在线| 久久久91精品国产一区二区精品| 在线看日韩av| 欧美偷拍一区二区| 欧美一区日韩一区| 亚洲国产精品久久久久秋霞不卡| 日韩亚洲欧美一区二区三区| 国产精品久久久对白| 久久狠狠久久综合桃花| 91久久香蕉国产日韩欧美9色| 亚洲综合99| 亚洲成人在线观看视频| 欧美日韩卡一卡二| 久久国产精品一区二区三区| 亚洲国产精品久久久久婷婷老年| 中文在线一区| 在线观看国产精品淫| 欧美午夜免费影院| 久久久久国产精品午夜一区| 日韩亚洲视频| 免费看亚洲片| 亚洲欧美在线看| 亚洲精品网址在线观看| 国产欧美一区二区精品性色| 欧美激情一区二区三区在线| 欧美在线观看视频| 一本大道av伊人久久综合| 老司机aⅴ在线精品导航| 一区二区三区四区五区在线| 精品91视频| 国产日韩欧美中文在线播放| 欧美日韩国产精品一区二区亚洲| 久久久久久久久岛国免费|