c#之如何安全的跨线程访问控件

      在窗体开发中用到线程是很常见的事情。比如一边下载东西一边点击其他功能。如果只有一个主线程。势必只会但同一时间处理一个功能,完了之后,才会去处理下一个功能。这样的体验,相比用一次,你就再也不想碰了。再比如,加入进度条控件的软件。进度条从0读取到100的过程中,是有一个while循环的。如果你放到主线程里面执行,那就呵呵了。在这个循环停止前,你会发现你的软件绝逼处于卡死状态。鼠标都拖不动它。为了解决这类问题。我们需要附加一个线程,相当于请一个助理,帮我们解决循环的事。这样我们的主线程才有闲工夫管理其他事。这样鼠标点击拖动窗体或者其他功能就不会受到影响。

        好了,让我们一起,学习一下,怎么使用线程,以及附加的自定义线程如何安全的跨线程访问主线程中的控件。这里说明一下,所有的控件都属于主线程的,任何外来线程访问它,比如给它的属性赋值操作。都要采取安全的方式访问。这是软件开发的原则。那么现在我们先来看看,怎么做,才是线程访问不安全的呢?

请看代码:

 
 public Form1()
        {
            InitializeComponent();
            Control.CheckForIllegalCrossThreadCalls = false;//禁止编译器对跨线程访问做检查,如果不加这一句,绝逼报错。不信你试试!
        }
        
private void button1_Click(object sender, EventArgs e)
        {
            Thread thread1 = new Thread(new ParameterizedThreadStart(UpdateLabel));
            thread1.Start("更新Label控件");
        }
        private void UpdateLabel(object str)
        {
            this.label1.Text = str.ToString();//这个线程调用了主线程中的label1控件呵呵
        }

以上就是不安全访问控件。如果你不加

Control.CheckForIllegalCrossThreadCalls = false;

运行是会报错的。编译器会对这种跨线程做安全检查。如果你这样直接访问,它是不允许的。所以必须禁用掉才可以的。当然咯,你禁用掉就可以正常运行了。但是这样不安全。做好不要这样使用咯。平时用的玩没关系。工作了这样用,显得很不专业咯。下面介绍安全的跨线程访问

很多网上这样写的。发现依然不行。虽然是创建了另外一个附加进程,但是它获取了主线程的控制权,尽管不会报错,倒是界面会卡死。

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);//使用begininvoke方法
        }
         private void ThreadFunction()
        {
              while (true)
            {
                this.textBox1.Text = DateTime.Now.ToString();
                Thread.Sleep(1000);
            }
        }
    }

下面来看看正确的跨线程姿势:

第一种方式:

 delegate void done();
        private Thread thread1 = null;

        private void Form1_Load(object sender, EventArgs e)
        {
            //thread1 = new Thread(new ThreadStart(counter));
            thread1 = new Thread(new ThreadStart(crossthread));
            thread1.IsBackground = true;
            thread1.Start();
        }

        private void crossthread() {
            while (true)
            {
                Thread.Sleep(1000);
                dowork();
            }

        }

        private void dowork() {
            if (this.txt_count.InvokeRequired) {
                done Done = new done(dowork);
                this.Invoke(Done);
            }
            else
            {
                    this.txt_count.Text = DateTime.Now.ToString();
            }
   
        }

近两天会更新更多的方式。。。暂时写到这么多。楼主要睡觉了嘻嘻

转载于:https://my.oschina.net/RabbitXiao/blog/534506

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值