WaitCallback如何传递ref参数

timelight2000 2015-01-16 01:08:49
按照MSDN推荐,线程中如果要传递参数并返回值,可把处理函数及数据封装进类里。
如果要在线程里改变某个主线程变量的值怎么办?
我感觉函数已经封装进类了,没法直接用主线程的全局变量,那就必须使用引用类型把变量传进类里。可这个类如何构造我就不明白了。

我的程序:

class MyProc
{
public void Calc(ref int count)
{
Interlocked.Increment(ref count);
Thread.Sleep(100);
}
}
class testref
{
private int count;
public testref()
{
count = 0;
}
public void test()
{
for (int i = 0; i < 10; i++)
{
Interlocked.Increment(ref count);
MyProc proc = new MyProc();
//就是不知道下一句里的第二个参数怎么构造才能从proc.Calc里引用count
ThreadPool.QueueUserWorkItem(new WaitCallback(proc.Calc), ref count);
}
MessageBox.Show("Count=" + count.ToString());
Thread.Sleep(2000);
MessageBox.Show("Count=" + count.ToString());
}
}

private void button2_Click(object sender, EventArgs e)
{
testref t = new testref();
t.test();
}
...全文
313 7 打赏 收藏 转发到动态 举报
写回复
用AI写文章
7 条回复
切换为时间正序
请发表友善的回复…
发表回复
timelight2000 2015-01-16
  • 打赏
  • 举报
回复
引用 3 楼 sp1234 的回复:
从你堆砌太多的问题代码来看,线程会让你给弄成累赘的。你需要学点简单化、傻瓜化的程序规范。 一个异步(不能保证被同步调用)的方法,它有东西要输出,它要使用委托回调来通知它要输出的结果,例如定义为
public void Calc(int input, Action<int> callback)
这样设计接口!
我确实没有足够经验,写的东西不规范,诚恳向专家请教。您说的这个道理我明白,但实际过程中是很复杂的。 我要做的东西是这样:testref是我这些功能的封装,每个实例可以做一类这样的工作。在testref里会维护一些myproc对象。testref.test()是每次调用的工作集,会不定时的多次从程序主进程调用;每次调用都会对所有的myproc对象分别调用一次myproc.calc();为了保证速度和主程序响应,所以使用了线程池。对于在testref里面实例化的每一个myproc对象,在myproc构造函数中会传入使其工作必需的参数。calc()执行会有多个结果,分别存在myproc的几个成员里,且其执行效果是累加的。也就是说以前执行的calc的结果会影响下一次执行calc。 我要从testref里随时查看目前执行的结果,这可以通过调用myproc的成员得到。而我要随时知道当前是否有工作正在进行,以便决定能否在testref对象里增加或减少myproc对象,这就需要在testref里保存一个变量count,但要从myproc里去维护它,增加一个线程运行时就+1,线程关闭就-1。这个变量也就是我这次遇到的问题,当然现在在大家帮助下已经能运行了。 但是在这样的目标下,程序结构设置成怎样才是合理的呢?
timelight2000 2015-01-16
  • 打赏
  • 举报
回复
引用 4 楼 sp1234 的回复:
我再给你纠正一个、可能是你的老师误导了你的说法,方法
void Calc(ref int x)
{
    x = 3;
}

void test()
{
    value = 0;
    Calc(ref value);
    Debug.Assert(value == 3);
}
这能说“方法 Calc 内部把外部变量value给改为3”了吗?如果你老是那样教你,就是非常明显在瞎说了!没有一个正规的说明敢说“ref的意思就是方法内部去修改方法外部的变量值”。如果是你自己瞎猜的,那么情有可原,实际情况完全不是这么回事。ref 的意思是说 Calc 执行结束之后会把结果赋值给 value,而绝不是说 Calc 内部会去修改外部变量 value。 高级语言不胡乱解释什么参数上的 ref 的作用,去掉了c语言的胡乱之源。这是各种高级语言跟 c 的重要区别。
感谢。我确实把ref理解成C里面的指针了,以为ref就是传了个指针过去,直接操作原来的内存位置呢。
timelight2000 2015-01-16
  • 打赏
  • 举报
回复
引用 1 楼 github_22161131 的回复:

class MyProc
{
    public void Calc(object state)
    {
        ((Action)state)();
        Thread.Sleep(100);
    }
}

class testref
{
    private int count;
    
    public void test()
    {
        for (int i = 0; i < 10; i++)
        {
            Interlocked.Increment(ref count);
            MyProc proc = new MyProc();
            ThreadPool.QueueUserWorkItem(new WaitCallback(proc.Calc), new Action(() => Interlocked.Increment(ref count)));
        }
        // ...
    }
}
非常感谢。第一种方法顺利执行,不过这第二种生成报错。Action不是具有一个参数的无返回值的函数的委托吗?这个语法好像有些问题啊。但是我不会改,因为感觉这里没有参数啊
  • 打赏
  • 举报
回复
我再给你纠正一个、可能是你的老师误导了你的说法,方法
void Calc(ref int x)
{
    x = 3;
}

void test()
{
    value = 0;
    Calc(ref value);
    Debug.Assert(value == 3);
}
这能说“方法 Calc 内部把外部变量value给改为3”了吗?如果你老是那样教你,就是非常明显在瞎说了!没有一个正规的说明敢说“ref的意思就是方法内部去修改方法外部的变量值”。如果是你自己瞎猜的,那么情有可原,实际情况完全不是这么回事。ref 的意思是说 Calc 执行结束之后会把结果赋值给 value,而绝不是说 Calc 内部会去修改外部变量 value。 高级语言不胡乱解释什么参数上的 ref 的作用,去掉了c语言的胡乱之源。这是各种高级语言跟 c 的重要区别。
  • 打赏
  • 举报
回复
从你堆砌太多的问题代码来看,线程会让你给弄成累赘的。你需要学点简单化、傻瓜化的程序规范。 一个异步(不能保证被同步调用)的方法,它有东西要输出,它要使用委托回调来通知它要输出的结果,例如定义为
public void Calc(int input, Action<int> callback)
这样设计接口!
  • 打赏
  • 举报
回复
引用 楼主 timelight2000 的回复:
按照MSDN推荐,线程中如果要传递参数并返回值,可把处理函数及数据封装进类里。 如果要在线程里改变某个主线程变量的值怎么办?
一个方法要返回值,就可以使用函数返回值规范。而如果其处理实际上可能异步执行,那么就用异步回调方式来返回。 如果不想把大家都搞“死得很难看”,就稍微遵循一点规范。不要瞎写什么 ref。
winnowc 2015-01-16
  • 打赏
  • 举报
回复
WaitCallback不能这样用。既然count是testref类的私有变量,那么不应该在其它类对它直接操作。可以改成下面这样:

class MyProc
{
    public void Calc(object state)
    {
        ((testref)state).IncreaseCount();
        Thread.Sleep(100);
    }
}

class testref
{
    private int count;
    
    public void test()
    {
        for (int i = 0; i < 10; i++)
        {
            IncreaseCount();
            MyProc proc = new MyProc();
            ThreadPool.QueueUserWorkItem(new WaitCallback(proc.Calc), this);
        }
        // ...
    }

    public void IncreaseCount()
    {
        Interlocked.Increment(ref count);
    }
}
或者:

class MyProc
{
    public void Calc(object state)
    {
        ((Action)state)();
        Thread.Sleep(100);
    }
}

class testref
{
    private int count;
    
    public void test()
    {
        for (int i = 0; i < 10; i++)
        {
            Interlocked.Increment(ref count);
            MyProc proc = new MyProc();
            ThreadPool.QueueUserWorkItem(new WaitCallback(proc.Calc), new Action(() => Interlocked.Increment(ref count)));
        }
        // ...
    }
}

110,539

社区成员

发帖
与我相关
我的任务
社区描述
.NET技术 C#
社区管理员
  • C#
  • Web++
  • by_封爱
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

让您成为最强悍的C#开发者

试试用AI创作助手写篇文章吧