4

我正在将“同步”代码(即使用 Windows 事件等到其他线程完成某事)重构为“异步”代码(使用委托来实现回调机制)。

在同步代码中,有时我需要在等待结束后使用局部变量。当这样的代码异步时,那些局部变量会丢失(回调处理程序无法访问它们)。我可以将它们存储为类属性,但感觉很浪费。

在 C++ 中,我std::bind用来解决这个问题。我只需将尽可能多的参数添加到回调处理程序所需的局部变量,并在调用异步方法时绑定它们。例如,假设异步方法回调接收到一个类型为的对象,而调用者使用了两个类型为和CallbackParam的局部变量。LocalALocalB

void AsyncClass::MethodWhichCallsAsyncMethod(){
    LocalA localVarA;
    LocalB localVarB;
    // OnAsyncMethodDone will need localVarA and localVarB, so we bind them
    AsyncMethod( std::bind( &AsyncClass::OnAsyncMethodDone, this, std::placeholders::_1, localVarA, localVarB ) );
}

void AsynClass::AsyncMethod( std::function<void(CallbackParam)> callback ){
    CallbackParam result;
    //Compute result...
    if( callback )
        callback( result );
}

void AsyncClass::OnAsyncMethodDone( CallbackParam p, LocalA a, LocalB b ){
   //Do whatever needs to be done
}

在 C# 和 VB.NET 中是否有某种等价物?使用代表或其他东西?

更新:为了完整起见,这是基于@lasseespeholt 回答的我的示例的 C# 等效项:

using System;

public class AsyncClass {

        public void MethodWhichCallsAsyncMethod() {
            var a = new LocalA();
            var b = new LocalB();
            //Anonymous callback handler (equivalent to AsyncClass::OnAsyncMethodDone)
            Action<CallbackParam> callback = result => {
                //Do what needs to be done; result, a and b can be accessed
            };
            AsyncMethod( callback );
        }

        private void AsyncMethod( Action<CallbackParam> callback ) {
            var result = new CallbackParam();
            //Compute result...
            if( callback != null )
                callback( result );
        }
}
4

1 回答 1

3

更新:这几乎肯定不应该使用。在 C# 中使用 async/await 关键字

你可以利用如下的闭包:

void MethodWhichCallsAsyncMethod()
{
    int foo = 1;

    AsyncCallback callback = result =>
    {
        Console.WriteLine(foo); // Access to foo
    };

    AsyncMethod(callback);
}

void AsyncMethod(AsyncCallback callback)
{
    IAsyncResult result = null; // Compute result
    callback(result);
}

编译器生成一个包含“foo”的类,所以你不用这种方法保存任何东西,但它很干净。

于 2013-04-05T19:25:54.850 回答