国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > C# > 正文

c#中多線程訪問winform控件的若干問題小結

2020-01-24 03:05:38
字體:
來源:轉載
供稿:網友

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

首先來看傳統方法:

復制代碼 代碼如下:

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

運行這段代碼,我們會看到系統拋出一個異常: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();
      }

加入這句代碼以后發現程序可以正常運行了。這句代碼就是說在這個類中我們不檢查跨線程的調用是否合法(如果沒有 加這句話運行也沒有異常,那么說明系統以及默認的采用了不檢查的方式)。然而,這種方法不可取。我們查看CheckForIllegalCrossThreadCalls 這個屬性的定義,就會發現它是一個static的,也就是說無論我們在項目的什么地方修改了這個值,他就會在全局起作用。而且像這種跨線程訪問是否存在異常,我們通常都會去檢查。如果項目中其他人修改了這個屬性,那么我們的方案就失敗了,我們要采取另外的方案。

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

復制代碼 代碼如下:

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);//調用代理
        }

        private void ThreadFuntion()
        {
            while (true)
            {
                this.textBox1.Text = DateTime.Now.ToString();
                Thread.Sleep(1000);
            }
        }
    }

使用這種方式我們可以看到跨線程訪問的異常沒有了。但是新問題出現了,界面沒有響應了。為什么會出現這個問題, 我們只是讓新開的線程無限循環刷新,理論上應該不會對主線程產生影響的。其實不然,這種方式其實相當于把這個新開的線程“注入”到了主控制線程中,它取得了主線程的控制。只要這個線程不返回,那么主線程將永遠都無法響應。就算新開的線程中不使用無限循環,使可以返回了。這種方式的使用多線程也失去了它本來的意義。

現在來讓我們看看推薦的解決方案:

復制代碼 代碼如下:

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和無限循環放在等待異步的外面
                Thread.Sleep(1000);
                ThreadFunction();
            }
        }

        private void ThreadFunction()
        {
            if (this.textBox1.InvokeRequired)//等待異步
            {
                FlushClient fc = new FlushClient(ThreadFunction);
                this.Invoke(fc);//通過代理調用刷新方法
            }
            else
            {
                this.textBox1.Text = DateTime.Now.ToString();
            }
        }
    }

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

對于深山老林提出的問題,我最近找到了更優的解決方案,利用了delegate的異步調用,大家可以看看:

復制代碼 代碼如下:

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(ThreadFunction);
             fc.BeginInvoke(null,null);
        }

        private void ThreadFunction()
        {
              while (true)
            {
                this.textBox1.Text = DateTime.Now.ToString();
                Thread.Sleep(1000);
            }
        }
    }

這種方法也可以直接簡化為(因為delegate的異步就是開了一個異步線程):

復制代碼 代碼如下:

public partial class Form1 : Form
    {
        private delegate void FlushClient();//代理
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
             FlushClient fc=new FlushClient(ThreadFunction);
             fc.BeginInvoke(null,null);
        }

         private void ThreadFunction()
        {
              while (true)
            {

                this.textBox1.Text = DateTime.Now.ToString();
                Thread.Sleep(1000);
            }
        }
    }

貧僧用飄柔反饋

while (true)和Thread.Sleep(1000);不要放在invoke里面去控制就行了

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 青海省| 墨玉县| 邳州市| 杂多县| 彝良县| 乃东县| 香格里拉县| 邻水| 集贤县| 旌德县| 和硕县| 抚州市| 六枝特区| 陈巴尔虎旗| 庆安县| 莒南县| 靖安县| 阿鲁科尔沁旗| 远安县| 汪清县| 确山县| 扬州市| 阿克苏市| 双牌县| 商都县| 柳江县| 石屏县| 营口市| 崇明县| 容城县| 会理县| 黑龙江省| 运城市| 正蓝旗| 镇安县| 平武县| 高阳县| 建昌县| 阿巴嘎旗| 临武县| 宜良县|