C#多线程中使用代理(委托)

  net2.0以后,微软加强了控件的安全机制,不允许跨线程操作。
    如果想跨线程操作控件的话要使用委托。
 
以前有过相关学习,后来既不清了,看了一遍之前写的文章,觉得太烂了,重新总结一下:
 
思路:在主线程中定义一个方法,这个方法用于主线程,然后定义一个委托,将委托指向这个方法(或者说讲方法绑定到委托上),其他线程调用委托。达到迂回的调用了主线程的方法,控制主线程的资源。
ps:这里之所以用“委托”这个词,不用“代理”,主要是感觉应用环境不同。线程A操作线程B的资源,但是不允许直接操作,那么A只好访问一个允许访问的对象,然后在委托这个对象对B的资源进行操作。站在这个角度“委托”更合适一点。
 
知识点: Invoke 、InvokeRequired、delegate
新建一个winform工程,添加一个按钮、一个进度条名字默认: button1、progressBar1    
例子一,无参数无返回值的方法:
1、添加一个代理:
 
private  delegate void myDelegate1();
2、为修改进度条创建一个方法:
private void SetProgressBar1()
        {
            progressBar1.Value = 0;
            progressBar1.Maximum = 2;
            this.progressBar1.Value = this.progressBar1.Value + 1;
            MessageBox.Show("SetProgressBar1");
        }
3、添加一个线程将要调用的方法:
private void TRun1()
        {
             //判断控件是否在本线程内
              if (!this.progressBar1.InvokeRequired)
              {
                  // MessageBox.Show("同一线程内");
                  Thread.Sleep(2000);
              }
            else
              {
                myDelegate1 md1 = new myDelegate1(SetProgressBar1);
                 //MessageBox.Show("不是同一个线程");
                Thread.Sleep(2000);
                Invoke(md1);                  
              }
        }
4、开启新线程调用:
private void button1_Click(object sender, EventArgs e)
        {
           // RunWithInvoke();        
             Thread thread = new Thread(new ThreadStart(TRun1));
            thread.Start();
            MessageBox.Show("使用多线程");
        }
结果:单击button1之后弹出提示框“使用多线程”,两秒钟之后弹出提示框“SetProgressBar1”并且进度条增长到50%位置。
执行过程:
1、先创建一个新线程并指向“TRun1”方法
2、“TRun1”方法判断控件是否属于该线程
3、不属于该线程,创建一个代理,并将代理指向“SetProgressBar1”方法
4、停顿2秒钟,Invoke代理,从而执行“SetProgressBar1”方法。(“SetProgressBar1”方法在主线程地址空间执行)
例子二,有参数的方法:
1、    创建代理
  private delegate void myDelegate2(string s);
 
2、创建方法
        //跟SetProgressBarValue委托相匹配的方法
        private void SetProgressBar2(string s)
        {
            progressBar1.Value = 0;
            progressBar1.Maximum = 2;
            this.progressBar1.Value = this.progressBar1.Value + 1;
            MessageBox.Show("SetProgressBar2 "+s);
        }
3、   添加线程的方法
  private void TRun2()
        {
            //判断控件是否在本线程内
            if (!this.progressBar1.InvokeRequired)
            {
                // MessageBox.Show("同一线程内");
                Thread.Sleep(2000);
            }
            else
            {
                myDelegate2 md2 = new myDelegate2(SetProgressBar2);
                //MessageBox.Show("不是同一个线程");
                Thread.Sleep(2000);
                Invoke(md2, "TRun2");
            }
        }
4、   新线程
   private void button1_Click(object sender, EventArgs e)
        {
            Thread thread = new Thread(new ThreadStart(TRun2));
            thread.Start();
            MessageBox.Show("使用多线程");
        }
例子三,有返回值的方法
 
1 、创建代理
 
private delegate string  myDelegate3(string s);
2、创建方法
 
private string SetProgressBar3(string s)
        {
            progressBar1.Value = 0;
            progressBar1.Maximum = 2;
            this.progressBar1.Value = this.progressBar1.Value + 1;
            MessageBox.Show("SetProgressBar3 " + s);
 
            return ("return SetProgressBar3 " + s);
        }
3、添加线程的方法
 
private void TRun3()
        {
            //判断控件是否在本线程内
            if (!this.progressBar1.InvokeRequired)
            {
                // MessageBox.Show("同一线程内");
                Thread.Sleep(2000);
            }
            else
            {
                myDelegate3 md3 = new myDelegate3(SetProgressBar3);
 
                //MessageBox.Show("不是同一个线程");
                Thread.Sleep(2000);
                MessageBox.Show ( Invoke(md3, "TRun3").ToString()); 
            }
        }
4、   新线程
 
private void button1_Click(object sender, EventArgs e)
        {
            Thread thread = new Thread(new ThreadStart(TRun3));
            thread.Start();
            MessageBox.Show("使用多线程");
        }
  • 5
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值