使用delegate给线程传递参数

第一次在博客园写文章。

最近遇到一个问题,用到了多线程,以前用的时候线程启动时不需要传递参数,可现在需要时却被难了一把。。

还是先说说delegate

delegateC#中的一种类型,它实际上是一个能够持有对某个方法的引用的类。与其它的类不同,delegate类能够拥有一个签名(signature),并且它只能持有与它的签名相匹配的方法的引用。它所实现的功能与C/C++中的函数指针十分相似。它允许你传递一个类A的方法m给另一个类B的对象,使得类B的对象能够调用这个方法m。但与函数指针相比,delegate有许多函数指针不具备的优点。首先,函数指针只能指向静态函数,而delegate既可以引用静态函数,又可以引用非静态成员函数。在引用非静态成员函数时,delegate不但保存了对此函数入口指针的引用,而且还保存了调用此函数的类实例的引用。其次,与函数指针相比,delegate是面向对象、类型安全、可靠的受控(managed)对象。也就是说,runtime能够保证delegate指向一个有效的方法,你无须担心delegate会指向无效地址或者越界地址。(摘自:gerbil

      这一段先放着吧。

那如何在线程启动时传递参数呢?

先定义一个线程启动时的执行的方法:

         static   void  count( int  i)
        
{
            Console.WriteLine(
"i={0", i);
        }

很简单的一个方法,显示传入int值。

启动线程的方法也很简单:

             int  i  =   4 ;
            
new  System.Threading.Thread((System.Threading.ThreadStart) delegate   { count(i); } ).Start();

类似的,还可以用到TimerThreadPool
      ThreadPool中的应用:

System.Threading.ThreadPool.QueueUserWorkItem((System.Threading.WaitCallback) delegate   { count(i); } );

System.Threading.Timer中的应用

            System.Threading.Timer threadTimer  =   new  System.Threading.Timer((System.Threading.TimerCallback) delegate   { count(i); } , null , 0 , 30000 );

System.Timers.Timer 中的应用

            System.Timers.Timer timersTimer  =   new  System.Timers.Timer( 30000 );
            timersTimer.AutoReset 
=   false ;
            timersTimer.Elapsed 
+=   new  System.Timers.ElapsedEventHandler( delegate   { count(i); } );
            timersTimer.Enabled 
=   true ;
但是下面这段代码将输出些什么呢?
             for  ( int  i  =   0 ; i  <   3 ; i ++ )
            
{
                System.Timers.Timer timersTimer 
= new System.Timers.Timer(30000);
                timersTimer.AutoReset 
= false;
                timersTimer.Elapsed 
+= new System.Timers.ElapsedEventHandler(delegate { count(i); });
                timersTimer.Enabled 
= true;
            }

            System.Threading.Thread.Sleep(
600000 );

从代码的顺序上看应该输出:
i=0
i=1
i=2
再来看看实际的输出:
i=3
i=3
i=3

出现这种问题的原因就是上面提到的,delegate指定的只是一个地址,方法的入口地址,不但如此,方法变量也是指定的一个地址,当delegate代码还未执行时,变量所指定的地址的值已经改变,所以传入给方法的值也已经改变。

解决方法也很简单:把变量i变为私有变量。

             for  ( int  i  =   0 ; i  <   3 ; i ++ )
            
{
                
int j = i;//在此处添了私有变量
                System.Timers.Timer timersTimer = new System.Timers.Timer(30000);
                timersTimer.AutoReset 
= false;
                timersTimer.Elapsed 
+= new System.Timers.ElapsedEventHandler(delegate { count(j); });
                timersTimer.Enabled 
= true;
            }

            System.Threading.Thread.Sleep(
600000 );


再来看看输出结果:
i=0
i=1
i=2

转载于:https://www.cnblogs.com/zsea/archive/2008/05/29/1209700.html

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值