亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

c#中多線(xiàn)程訪(fǎng)問(wèn)winform控件的若干問(wèn)題小結(jié)

 更新時(shí)間:2013年10月31日 22:13:27   作者:  
大部分情況下都會(huì)碰到使用多線(xiàn)程控制界面上控件信息的問(wèn)題。然而我們并不能用傳統(tǒng)方法來(lái)解決這個(gè)問(wèn)題,下面我將詳細(xì)的介紹

我們?cè)谧鰓inform應(yīng)用的時(shí)候,大部分情況下都會(huì)碰到使用多線(xiàn)程控制界面上控件信息的問(wèn)題。然而我們并不能用傳統(tǒng)方法來(lái)解決這個(gè)問(wèn)題,下面我將詳細(xì)的介紹。

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

復(fù)制代碼 代碼如下:

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

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

      第一種方案,我們?cè)贔orm1_Load()方法中加一句代碼:

復(fù)制代碼 代碼如下:

private void Form1_Load(object sender, EventArgs e)
      {
            Control.CheckForIllegalCrossThreadCalls = false;
            Thread thread = new Thread(ThreadFuntion);
            thread.IsBackground = true;
            thread.Start();
      }

加入這句代碼以后發(fā)現(xiàn)程序可以正常運(yùn)行了。這句代碼就是說(shuō)在這個(gè)類(lèi)中我們不檢查跨線(xiàn)程的調(diào)用是否合法(如果沒(méi)有 加這句話(huà)運(yùn)行也沒(méi)有異常,那么說(shuō)明系統(tǒng)以及默認(rèn)的采用了不檢查的方式)。然而,這種方法不可取。我們查看CheckForIllegalCrossThreadCalls 這個(gè)屬性的定義,就會(huì)發(fā)現(xiàn)它是一個(gè)static的,也就是說(shuō)無(wú)論我們?cè)陧?xiàng)目的什么地方修改了這個(gè)值,他就會(huì)在全局起作用。而且像這種跨線(xiàn)程訪(fǎng)問(wèn)是否存在異常,我們通常都會(huì)去檢查。如果項(xiàng)目中其他人修改了這個(gè)屬性,那么我們的方案就失敗了,我們要采取另外的方案。

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

復(fù)制代碼 代碼如下:

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)程訪(fǎng)問(wèn)的異常沒(méi)有了。但是新問(wèn)題出現(xiàn)了,界面沒(méi)有響應(yīng)了。為什么會(huì)出現(xiàn)這個(gè)問(wèn)題, 我們只是讓新開(kāi)的線(xiàn)程無(wú)限循環(huán)刷新,理論上應(yīng)該不會(huì)對(duì)主線(xiàn)程產(chǎn)生影響的。其實(shí)不然,這種方式其實(shí)相當(dāng)于把這個(gè)新開(kāi)的線(xiàn)程“注入”到了主控制線(xiàn)程中,它取得了主線(xiàn)程的控制。只要這個(gè)線(xiàn)程不返回,那么主線(xiàn)程將永遠(yuǎn)都無(wú)法響應(yīng)。就算新開(kāi)的線(xiàn)程中不使用無(wú)限循環(huán),使可以返回了。這種方式的使用多線(xiàn)程也失去了它本來(lái)的意義。

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

復(fù)制代碼 代碼如下:

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

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

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

對(duì)于深山老林提出的問(wèn)題,我最近找到了更優(yōu)的解決方案,利用了delegate的異步調(diào)用,大家可以看看:

復(fù)制代碼 代碼如下:

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

這種方法也可以直接簡(jiǎn)化為(因?yàn)閐elegate的異步就是開(kāi)了一個(gè)異步線(xiàn)程):

復(fù)制代碼 代碼如下:

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里面去控制就行了

相關(guān)文章

最新評(píng)論