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

C#中跨線程訪問(wèn)控件問(wèn)題解決方案分享

 更新時(shí)間:2013年11月27日 16:59:56   作者:  
這篇文章主要介紹了C#中跨線程訪問(wèn)控件問(wèn)題解決方案,有需要的朋友可以參考一下

net 原則上禁止跨線程訪問(wèn)控件,因?yàn)檫@樣可能造成錯(cuò)誤的發(fā)生,推薦的解決方法是采用代理,用代理方法來(lái)間接操作不是同一線程創(chuàng)建的控件。

第二種方法是禁止編譯器對(duì)跨線程訪問(wèn)作檢查,可以實(shí)現(xiàn)訪問(wèn),但是出不出錯(cuò)不敢保證Control.CheckForIllegalCrossThreadCalls = false;

最近我在做一個(gè)項(xiàng)目,遇到了跨線程要去訪問(wèn)頁(yè)面控件.但是總是提示出錯(cuò),不能在其它線程中修改創(chuàng)建控件的線程的控件的值,后來(lái)采用了匿名代理,結(jié)果很輕松地解決了.解決過(guò)程如下:
首先在窗體上,創(chuàng)建一個(gè)listbox,lable.

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

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;
using System.Threading;

namespace AccessControl
{
    public partial class Form1 : Form
    {     
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        { 
            Thread newthread = new Thread(new ThreadStart(BackgroundProcess));
            newthread.Start();        

        }

        /// <summary>
        /// 定義一個(gè)代理
        /// </summary>
        private delegate void CrossThreadOperationControl();

        private void BackgroundProcess()
        {
            // 將代理實(shí)例化為一個(gè)匿名代理
            CrossThreadOperationControl CrossDelete = delegate()         
            {           
                int i = 1;
                while (i<5)
                {
                   // 向列表框增加一個(gè)項(xiàng)目
                    listBox1.Items.Add("Item " + i.ToString());                  
                    i++;
                }
                label1.Text = "我在新線程里訪問(wèn)這個(gè)lable!";
                listBox1.Items.Add(label1.Text);
            }  ;
            listBox1.Invoke(CrossDelete);          
        }      

    }
}

希望這個(gè)小技巧能夠?qū)δ愕牡膶W(xué)習(xí)和工作有所幫助.若有更好的辦法來(lái)解決跨線程訪問(wèn)控件的問(wèn)題,不防也拿出來(lái)大家分享一下.

C#跨線程訪問(wèn)控件運(yùn)行時(shí)錯(cuò)誤,使用MethodInvoker即可解決:

原代碼:

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

private void btnOK_Click(object sender, EventArgs e)
        {
            tslInfo.Text = "請(qǐng)稍候...";


            Thread td = new Thread(new ThreadStart(run));
            td.Start();
        }


        /// <summary>
        /// 線程方法
        /// </summary>
        private void run()
        {
            this.tslInfo.Text = "就緒";
        }


修改后:
復(fù)制代碼 代碼如下:

private void btnOK_Click(object sender, EventArgs e)
        {
            tslInfo.Text = "請(qǐng)稍候...";


            Thread td = new Thread(new ThreadStart(threadRun));
            td.Start();
        }


        /// <summary>
        /// 原線程方法
        /// </summary>
        private void run()
        {
            this.tslInfo.Text = "就緒";
        }

        /// <summary>
        /// 線程方法
        /// </summary>
        private void threadRun()
        {
            MethodInvoker In = new MethodInvoker(run);
            this.BeginInvoke(In);
        }


我們?cè)谧鰓inform應(yīng)用的時(shí)候,大部分情況下都會(huì)碰到使用多線程控制界面上控件信息的問(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中直接跨線程訪問(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è)類中我們不檢查跨線程的調(diào)用是否合法(如果沒(méi)有加這句話運(yùn)行也沒(méi)有異常,那么說(shuō)明系統(tǒng)以及默認(rèn)的采用了不檢查的方式)。然而,這種方法不可取。我們查看CheckForIllegalCrossThreadCalls 這個(gè)屬性的定義,就會(huì)發(fā)現(xiàn)它是一個(gè)static的,也就是說(shuō)無(wú)論我們?cè)陧?xiàng)目的什么地方修改了這個(gè)值,他就會(huì)在全局起作用。而且像這種跨線程訪問(wèn)是否存在異常,我們通常都會(huì)去檢查。如果項(xiàng)目中其他人修改了這個(gè)屬性,那么我們的方案就失敗了,我們要采取另外的方案。

      下面來(lái)看第二種方案,就是使用delegate和invoke來(lái)從其他線程中控制控件信息。網(wǎng)上有很多人寫了這種控制方式,然而我看了很多這種帖子,表明上看來(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);
            }
        }
    }

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

       對(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=new FlushClient(ThreadFunction);

             FlushClient.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è)異步線程):
復(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);
             FlushClient=new FlushClient(ThreadFunction);

             FlushClient.BeginInvoke(null,null);
        }

         private void ThreadFunction()
        {

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

        }
    }

相關(guān)文章

  • opencvsharp瑕疵檢測(cè)的實(shí)現(xiàn)示例

    opencvsharp瑕疵檢測(cè)的實(shí)現(xiàn)示例

    本文主要介紹了opencvsharp瑕疵檢測(cè)的實(shí)現(xiàn)示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2022-05-05
  • C#開(kāi)發(fā)答題贏錢游戲(自動(dòng)答題器)

    C#開(kāi)發(fā)答題贏錢游戲(自動(dòng)答題器)

    現(xiàn)在最火的直播游戲,那就是答題贏錢直播了,如百萬(wàn)英雄、芝士超人、花椒直播、沖頂大會(huì)等等,這些游戲的玩法都很簡(jiǎn)單,答對(duì)12題即可瓜分獎(jiǎng)金了。玩法雖簡(jiǎn)單但是完全答對(duì)12題難度就挺高了,下面小編給大家?guī)?lái)了C#開(kāi)發(fā)答題贏錢游戲,需要的朋友參考下吧
    2018-01-01
  • C#特性 迭代器(下) yield以及流的延遲計(jì)算

    C#特性 迭代器(下) yield以及流的延遲計(jì)算

    這篇文章主要介紹了C#特性 迭代器(下) yield以及流的延遲計(jì)算,需要的朋友可以參考下
    2014-12-12
  • C#實(shí)現(xiàn)文字轉(zhuǎn)語(yǔ)音功能

    C#實(shí)現(xiàn)文字轉(zhuǎn)語(yǔ)音功能

    這篇文章主要為大家詳細(xì)介紹了C#實(shí)現(xiàn)文字轉(zhuǎn)語(yǔ)音功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-03-03
  • Unity實(shí)現(xiàn)毫秒延時(shí)回調(diào)功能

    Unity實(shí)現(xiàn)毫秒延時(shí)回調(diào)功能

    這篇文章主要為大家詳細(xì)介紹了Unity實(shí)現(xiàn)毫秒延時(shí)回調(diào)功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-09-09
  • C#中子類調(diào)用父類的實(shí)現(xiàn)方法

    C#中子類調(diào)用父類的實(shí)現(xiàn)方法

    這篇文章主要介紹了C#中子類調(diào)用父類的實(shí)現(xiàn)方法,通過(guò)實(shí)例逐步分析了類中初始化構(gòu)造函數(shù)的執(zhí)行順序問(wèn)題,有助于加深對(duì)C#面向?qū)ο蟪绦蛟O(shè)計(jì)的理解,需要的朋友可以參考下
    2014-09-09
  • C#中創(chuàng)建PDF網(wǎng)格并插入圖片的方法

    C#中創(chuàng)建PDF網(wǎng)格并插入圖片的方法

    這篇文章我將向大家演示如何以編程的方式在PDF文檔中創(chuàng)建一個(gè)網(wǎng)格,并將圖片插入特定的網(wǎng)格中。對(duì)c# pdf 網(wǎng)格 插入圖片的知識(shí)感興趣的朋友一起看看吧
    2016-11-11
  • 淺析.NET中AsyncLocal的實(shí)現(xiàn)原理

    淺析.NET中AsyncLocal的實(shí)現(xiàn)原理

    這篇文章主要為大家詳細(xì)介紹了.NET中AsyncLocal的具體實(shí)現(xiàn)原理,文中的示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,如果有講得不清晰或不準(zhǔn)確的地方,還望指出
    2023-08-08
  • Windows服務(wù)編寫(Windows Service,system權(quán)限)程序顯示界面與用戶交互(xp,win7通用)

    Windows服務(wù)編寫(Windows Service,system權(quán)限)程序顯示界面與用戶交互(xp,win7通用)

    這篇文章主要介紹了Windows服務(wù)編寫Windows Service,system權(quán)限程序顯示界面與用戶交互(xp,win7通用) ,需要的朋友可以參考下
    2016-04-04
  • C# 中的List.Sort()--集合排序方法全面解析

    C# 中的List.Sort()--集合排序方法全面解析

    這篇文章主要介紹了C# 中的List.Sort()--集合排序方法全面解析,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2021-01-01

最新評(píng)論