Как заставить асинхронный метод возвращать значение?

Я знаю, как сделать асинхронные методы, но сказать, что у меня есть метод, который делает много работы, а затем возвращает логическое значение?

Как вернуть логическое значение при обратном вызове?

уточнение:

public bool Foo(){
    Thread.Sleep(100000); // Do work
    return true;
}

Я хочу иметь возможность сделать это асинхронным.

6 ответов


есть несколько способов сделать это... самое простое - чтобы асинхронный метод также выполнял последующую операцию. Другим популярным подходом является передача обратного вызова, т. е.

void RunFooAsync(..., Action<bool> callback) {
     // do some stuff
     bool result = ...

     if(callback != null) callback(result);
}

другим подходом было бы вызвать событие (с результатом в данных event-args), когда асинхронная операция завершена.

кроме того, если вы используете TPL, вы можете использовать ContinueWith:

Task<bool> outerTask = ...;
outerTask.ContinueWith(task =>
{
    bool result = task.Result;
    // do something with that
});

С C# 5.0, вы можете указать способ, как

public async Task<bool> doAsyncOperation()
{
    // do work
    return true;
}

bool result = await doAsyncOperation();

вероятно, самый простой способ сделать это-создать делегат, а затем BeginInvoke, следует подождать какое-то время в будущем, и EndInvoke.

public bool Foo(){
    Thread.Sleep(100000); // Do work
    return true;
}

public SomeMethod()
{
    var fooCaller = new Func<bool>(Foo);
    // Call the method asynchronously
    var asyncResult = fooCaller.BeginInvoke(null, null);

    // Potentially do other work while the asynchronous method is executing.

    // Finally, wait for result
    asyncResult.AsyncWaitHandle.WaitOne();
    bool fooResult = fooCaller.EndInvoke(asyncResult);

    Console.WriteLine("Foo returned {0}", fooResult);
}

используйте BackgroundWorker. Это позволит вам получить обратные вызовы по завершении и позволит вам отслеживать прогресс. Можно задать результирующее значение аргументов события для результирующего значения.

    public void UseBackgroundWorker()
    {
        var worker = new BackgroundWorker();
        worker.DoWork += DoWork;
        worker.RunWorkerCompleted += WorkDone;
        worker.RunWorkerAsync("input");
    }

    public void DoWork(object sender, DoWorkEventArgs e)
    {
        e.Result = e.Argument.Equals("input");
        Thread.Sleep(1000);
    }

    public void WorkDone(object sender, RunWorkerCompletedEventArgs e)
    {
        var result = (bool) e.Result;
    }

возможно, вы можете попытаться начать вызывать делегат, указывающий на ваш метод так:



    delegate string SynchOperation(string value);

    class Program
    {
        static void Main(string[] args)
        {
            BeginTheSynchronousOperation(CallbackOperation, "my value");
            Console.ReadLine();
        }

        static void BeginTheSynchronousOperation(AsyncCallback callback, string value)
        {
            SynchOperation op = new SynchOperation(SynchronousOperation);
            op.BeginInvoke(value, callback, op);
        }

        static string SynchronousOperation(string value)
        {
            Thread.Sleep(10000);
            return value;
        }

        static void CallbackOperation(IAsyncResult result)
        {
            // get your delegate
            var ar = result.AsyncState as SynchOperation;
            // end invoke and get value
            var returned = ar.EndInvoke(result);

            Console.WriteLine(returned);
        }
    }

затем используйте значение в методе, отправленном как AsyncCallback для продолжения..


вы должны использовать EndXXX вашего асинхронного метода для возврата значения. EndXXX должен ждать, пока не появится результат, используя WaitHandle IAsyncResult и чем вернуться со значением.