乡下人产国偷v产偷v自拍,国产午夜片在线观看,婷婷成人亚洲综合国产麻豆,久久综合给合久久狠狠狠9

  • <output id="e9wm2"></output>
    <s id="e9wm2"><nobr id="e9wm2"><ins id="e9wm2"></ins></nobr></s>

    • 分享

      c#中使用多線程訪問winform中控件

       orion360doc 2011-01-25
      我們在做winform應用的時候,大部分情況下都會碰到使用多線程控制界面上控件信息的問題。然而我們并不能用傳統(tǒng)方法來做這個問題,下面我將詳細的介紹。

            首先來看傳統(tǒng)方法:

            public partial class Form1 : Form
          {
              public Form1()
              {
                  InitializeComponent();
              }
              private void Form1_Load(object sender, EventArgs e)
              {
                  Thread thread = new Thread(ThreadFuntion);
                  thread.IsBackground = true;
                  thread.Start();
              }
              private void ThreadFuntion()
              {
                  while (true)
                  {
                      this.textBox1.Text = DateTime.Now.ToString();
                      Thread.Sleep(1000);
                  }
              }
          }

             運行這段代碼,我們會看到系統(tǒng)拋出一個異常:Cross-thread operation not valid:Control 'textBox1' accessed from a thread other than the thread it was created on . 這是因為.net 2.0以后加強了安全機制,不允許在winform中直接跨線程訪問控件的屬性。那么怎么解決這個問題呢,下面提供幾種方案。

            第一種方案,我們在Form1_Load()方法中加一句代碼:

            private void Form1_Load(object sender, EventArgs e)
            {
                  Control.CheckForIllegalCrossThreadCalls = false;
                  Thread thread = new Thread(ThreadFuntion);
                  thread.IsBackground = true;
                  thread.Start();
              }
            加入這句代碼以后發(fā)現(xiàn)程序可以正常運行了。這句代碼就是說在這個類中我們不檢查跨線程的調(diào)用是否合法(如果沒有加這句話運行也沒有異常,那么說明系統(tǒng)以及默認的采用了不檢查的方式)。然而,這種方法不可取。我們查看CheckForIllegalCrossThreadCalls 這個屬性的定義,就會發(fā)現(xiàn)它是一個static的,也就是說無論我們在項目的什么地方修改了這個值,他就會在全局起作用。而且像這種跨線程訪問是否存在異常,我們通常都會去檢查。如果項目中其他人修改了這個屬性,那么我們的方案就失敗了,我們要采取另外的方案。

            下面來看第二種方案,就是使用delegate和invoke來從其他線程中控制控件信息。網(wǎng)上有很多人寫了這種控制方式,然而我看了很多這種帖子,表明上看來是沒有什么問題的,但是實際上并沒有解決這個問題,首先來看網(wǎng)絡上的那種不完善的方式:

      public partial class Form1 : Form
          {
              private delegate void FlushClient();//代理
              public Form1()
              {
                  InitializeComponent();
              }
              private void Form1_Load(object sender, EventArgs e)
              {
                  Thread thread = new Thread(CrossThreadFlush);

                  thread.IsBackground=true;
                  thread.Start();
              }

              private void CrossThreadFlush()
              {
                  //將代理綁定到方法
                  FlushClient fc = new FlushClient(ThreadFuntion);
                  this.BeginInvoke(fc);//調(diào)用代理
              }
              private void ThreadFuntion()
              {
                  while (true)
                  {
                      this.textBox1.Text = DateTime.Now.ToString();
                      Thread.Sleep(1000);
                  }
              }
          }

             使用這種方式我們可以看到跨線程訪問的異常沒有了。但是新問題出現(xiàn)了,界面沒有響應了。為什么會出現(xiàn)這個問題,我們只是讓新開的線程無限循環(huán)刷新,理論上應該不會對主線程產(chǎn)生影響的。

      其實不然,這種方式其實相當于把這個新開的線程“注入”到了主控制線程中,它取得了主線程的控制。只要這個線程不返回,那么主線程將永遠都無法響應。就算新開的線程中不使用無限循環(huán),使可以返回了。這種方式的使用多線程也失去了它本來的意義。

             現(xiàn)在來讓我們看看推薦的解決方案:

      public partial class Form1 : Form
          {
              private delegate void FlushClient();//代理
              public Form1()
              {
                  InitializeComponent();
              }
              private void Form1_Load(object sender, EventArgs e)
              {
                  Thread thread = new Thread(CrossThreadFlush);
                  thread.IsBackground = true;
                  thread.Start();
              }

              private void CrossThreadFlush()
              {
                  while (true)
                  {
                      //將sleep和無限循環(huán)放在等待異步的外面
                      Thread.Sleep(1000);
                      ThreadFunction();
                  }
              }
              private void ThreadFunction()
              {
                  if (this.textBox1.InvokeRequired)//等待異步
                  {
                      FlushClient fc = new FlushClient(ThreadFunction);
                      this.Invoke(fc);//通過代理調(diào)用刷新方法
                  }
                  else
                  {
                      this.textBox1.Text = DateTime.Now.ToString();
                  }
              }
          }

             運行上述代碼,我們可以看到問題已經(jīng)被解決了,通過等待異步,我們就不會總是持有主線程的控制,這樣就可以在不發(fā)生跨線程調(diào)用異常的情況下完成多線程對winform多線程控件的控制了。

        本站是提供個人知識管理的網(wǎng)絡存儲空間,所有內(nèi)容均由用戶發(fā)布,不代表本站觀點。請注意甄別內(nèi)容中的聯(lián)系方式、誘導購買等信息,謹防詐騙。如發(fā)現(xiàn)有害或侵權內(nèi)容,請點擊一鍵舉報。
        轉藏 分享 獻花(0

        0條評論

        發(fā)表

        請遵守用戶 評論公約

        類似文章 更多