How do I call async methods in asp.net C# 4.0?

14,459

Solution 1

Have a look at using Tasks, this was available in .Net 4 and should help you. A simple example might look like this:

public void MainFlow()
{
   Task taskWork = Task.Factory.StartNew(new Action(DoWork));
   //Do other work
   //Then wait for thread finish
   taskWork.Wait();
}


private void DoWork()
{
   //Do work
}

For more, have a look here

Solution 2

Use the Task.ContinueWith which basically executes your action once the task completes.

Some examples which could be useful:

http://msdn.microsoft.com/en-us/library/dd784422.aspx

Solution 3

We made some tool class for executing async operations.

What these methods really do: execute action asynchronously and then posts the execution of the completition accordingly to the synchronization context.

We made this as the wrapper of the TPL to easily use it iside WPF when we call long running operations. This class also has analogues for actions which can be cancelled, not just abandoned.

public static class AsyncExecutor
    {
        public static CancellationTokenSource ExecuteBlockingOperation(Action action, Action completition, Action<AggregateException> onException)
        {
            if (action == null)
                throw new ArgumentNullException("action");

            if (completition == null)
                throw new ArgumentNullException("completition");

            var cts = new CancellationTokenSource();
            var token = cts.Token;

            var scheduler = TaskScheduler.FromCurrentSynchronizationContext();

            var task = new Task(action, TaskCreationOptions.LongRunning);
            task.ContinueWith(asyncPartTask => onException(asyncPartTask.Exception), CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, scheduler);
            task.ContinueWith(asyncPart =>
                                  {
                                      if (!token.IsCancellationRequested)
                                          completition();
                                  },
                              CancellationToken.None, TaskContinuationOptions.OnlyOnRanToCompletion, scheduler);
            task.Start();

            return cts;

        }

        public static CancellationTokenSource ExecuteBlockingOperation<TIn>(Action<TIn> action, TIn parameter, Action<TIn> completition, Action<AggregateException, TIn> onException)
        {
            if (action == null)
                throw new ArgumentNullException("action");

            if (completition == null)
                throw new ArgumentNullException("completition");

            var cts = new CancellationTokenSource();
            var token = cts.Token;

            var scheduler = TaskScheduler.FromCurrentSynchronizationContext();

            var task = new Task(() => action(parameter), TaskCreationOptions.LongRunning);
            task.ContinueWith(asyncPartTask => onException(asyncPartTask.Exception, parameter), CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, scheduler);
            task.ContinueWith(asyncPart =>
                                  {
                                      if (!token.IsCancellationRequested)
                                          completition(parameter);
                                  },
                              CancellationToken.None, TaskContinuationOptions.OnlyOnRanToCompletion, scheduler);
            task.Start();

            return cts;

        }

        public static CancellationTokenSource ExecuteBlockingOperation<TOut>(Func<TOut> func, Action<TOut> completition, Action<AggregateException> onException)
        {
            if (func == null)
                throw new ArgumentNullException("func");

            if (completition == null)
                throw new ArgumentNullException("completition");

            var cts = new CancellationTokenSource();
            var token = cts.Token;

            var scheduler = TaskScheduler.FromCurrentSynchronizationContext();

            var task = new Task<TOut>(func, TaskCreationOptions.LongRunning);
            task.ContinueWith(asyncPartTask => onException(asyncPartTask.Exception), CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, scheduler);
            task.ContinueWith(asyncPart =>
                                  {
                                      if (!token.IsCancellationRequested)
                                          completition(asyncPart.Result);
                                  },
                              CancellationToken.None, TaskContinuationOptions.OnlyOnRanToCompletion, scheduler);
            task.Start();

            return cts;

        }

        public static CancellationTokenSource ExecuteBlockingOperation<TIn, TOut>(Func<TIn, TOut> func, TIn parameter, Action<TOut, TIn> completition, Action<AggregateException, TIn> onException)
        {
            if (func == null)
                throw new ArgumentNullException("func");

            if (completition == null)
                throw new ArgumentNullException("completition");

            var cts = new CancellationTokenSource();
            var token = cts.Token;

            var scheduler = TaskScheduler.FromCurrentSynchronizationContext();

            var task = new Task<TOut>(() => func(parameter), TaskCreationOptions.LongRunning);
            task.ContinueWith(asyncPartTask => onException(asyncPartTask.Exception, parameter), CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, scheduler);
            task.ContinueWith(asyncPart =>
            {
                if (!token.IsCancellationRequested)
                    completition(asyncPart.Result, parameter);
            },
                              CancellationToken.None, TaskContinuationOptions.OnlyOnRanToCompletion, scheduler);
            task.Start();

            return cts;
        }

        public static CancellationTokenSource ExecuteBlockingOperation<TIn, TOut>(Func<TIn, TOut> func, TIn parameter, Action<TOut> completition, Action<AggregateException, TIn> onException)
        {
            if (func == null)
                throw new ArgumentNullException("func");

            if (completition == null)
                throw new ArgumentNullException("completition");

            var cts = new CancellationTokenSource();
            var token = cts.Token;

            var scheduler = TaskScheduler.FromCurrentSynchronizationContext();

            var task = new Task<TOut>(() => func(parameter), TaskCreationOptions.LongRunning);
            task.ContinueWith(asyncPartTask => onException(asyncPartTask.Exception, parameter), CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, scheduler);
            task.ContinueWith(asyncPart =>
            {
                if (!token.IsCancellationRequested)
                    completition(asyncPart.Result);
            },
                              CancellationToken.None, TaskContinuationOptions.OnlyOnRanToCompletion, scheduler);
            task.Start();

            return cts;
        }

        public static CancellationTokenSource ExecuteBlockingOperation<TIn, TOut>(Func<TIn, TOut> func, TIn parameter, Action<TIn> completition, Action<AggregateException, TIn> onException)
        {
            if (func == null)
                throw new ArgumentNullException("func");

            if (completition == null)
                throw new ArgumentNullException("completition");

            var cts = new CancellationTokenSource();
            var token = cts.Token;

            var scheduler = TaskScheduler.FromCurrentSynchronizationContext();

            var task = new Task<TOut>(() => func(parameter), TaskCreationOptions.LongRunning);
            task.ContinueWith(asyncPartTask => onException(asyncPartTask.Exception, parameter), CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, scheduler);
            task.ContinueWith(asyncPart =>
            {
                if (!token.IsCancellationRequested)
                    completition(parameter);
            },
                              CancellationToken.None, TaskContinuationOptions.OnlyOnRanToCompletion, scheduler);
            task.Start();

            return cts;
        }

        public static CancellationTokenSource ExecuteBlockingOperation<TIn, TOut>(Func<TIn, TOut> func, TIn parameter, Action completition, Action<AggregateException, TIn> onException)
        {
            if (func == null)
                throw new ArgumentNullException("func");

            if (completition == null)
                throw new ArgumentNullException("completition");

            var cts = new CancellationTokenSource();
            var token = cts.Token;

            var scheduler = TaskScheduler.FromCurrentSynchronizationContext();

            var task = new Task<TOut>(() => func(parameter), TaskCreationOptions.LongRunning);
            task.ContinueWith(asyncPartTask => onException(asyncPartTask.Exception, parameter), CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, scheduler);
            task.ContinueWith(asyncPart =>
            {
                if (!token.IsCancellationRequested)
                    completition();
            },
                              CancellationToken.None, TaskContinuationOptions.OnlyOnRanToCompletion, scheduler);
            task.Start();

            return cts;
        }

        public static CancellationTokenSource ExecuteBlockingOperation<TIn, TOut>(Func<TIn, TOut> func, TIn parameter, Action<TIn> completition, Action<AggregateException> onException)
        {
            if (func == null)
                throw new ArgumentNullException("func");

            if (completition == null)
                throw new ArgumentNullException("completition");

            var cts = new CancellationTokenSource();
            var token = cts.Token;

            var scheduler = TaskScheduler.FromCurrentSynchronizationContext();

            var task = new Task<TOut>(() => func(parameter), TaskCreationOptions.LongRunning);
            task.ContinueWith(asyncPartTask => onException(asyncPartTask.Exception), CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, scheduler);
            task.ContinueWith(asyncPart =>
            {
                if (!token.IsCancellationRequested)
                    completition(parameter);
            },
                              CancellationToken.None, TaskContinuationOptions.OnlyOnRanToCompletion, scheduler);
            task.Start();

            return cts;
        }

        public static CancellationTokenSource ExecuteBlockingOperation<TIn, TOut>(Func<TIn, TOut> func, TIn parameter, Action<TOut, TIn> completition, Action<AggregateException> onException)
        {
            if (func == null)
                throw new ArgumentNullException("func");

            if (completition == null)
                throw new ArgumentNullException("completition");

            var cts = new CancellationTokenSource();
            var token = cts.Token;

            var scheduler = TaskScheduler.FromCurrentSynchronizationContext();

            var task = new Task<TOut>(() => func(parameter), TaskCreationOptions.LongRunning);
            task.ContinueWith(asyncPartTask => onException(asyncPartTask.Exception), CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, scheduler);
            task.ContinueWith(asyncPart =>
            {
                if (!token.IsCancellationRequested)
                    completition(asyncPart.Result, parameter);
            },
                              CancellationToken.None, TaskContinuationOptions.OnlyOnRanToCompletion, scheduler);
            task.Start();

            return cts;
        }

        public static CancellationTokenSource ExecuteBlockingOperation<TIn, TOut>(Func<TIn, TOut> func, TIn parameter, Action<TOut> completition, Action<AggregateException> onException)
        {
            if (func == null)
                throw new ArgumentNullException("func");

            if (completition == null)
                throw new ArgumentNullException("completition");

            var cts = new CancellationTokenSource();
            var token = cts.Token;

            var scheduler = TaskScheduler.FromCurrentSynchronizationContext();

            var task = new Task<TOut>(() => func(parameter), TaskCreationOptions.LongRunning);
            task.ContinueWith(asyncPartTask => onException(asyncPartTask.Exception), CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, scheduler);
            task.ContinueWith(asyncPart =>
            {
                if (!token.IsCancellationRequested)
                    completition(asyncPart.Result);
            },
                              CancellationToken.None, TaskContinuationOptions.OnlyOnRanToCompletion, scheduler);
            task.Start();

            return cts;
        }

        public static CancellationTokenSource ExecuteBlockingOperation<TIn, TOut>(Func<TIn, TOut> func, TIn parameter, Action completition, Action<AggregateException> onException)
        {
            if (func == null)
                throw new ArgumentNullException("func");

            if (completition == null)
                throw new ArgumentNullException("completition");

            var cts = new CancellationTokenSource();
            var token = cts.Token;

            var scheduler = TaskScheduler.FromCurrentSynchronizationContext();

            var task = new Task<TOut>(() => func(parameter), TaskCreationOptions.LongRunning);
            task.ContinueWith(asyncPartTask => onException(asyncPartTask.Exception), CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, scheduler);
            task.ContinueWith(asyncPart =>
            {
                if (!token.IsCancellationRequested)
                    completition();
            },
                              CancellationToken.None, TaskContinuationOptions.OnlyOnRanToCompletion, scheduler);
            task.Start();

            return cts;
        }

        public static void ExecuteBlockingOperation(Action action, Action completition, Func<bool> shouldComplete, Action<AggregateException> onException)
        {
            if (action == null)
                throw new ArgumentNullException("action");

            if (completition == null)
                throw new ArgumentNullException("completition");

            var scheduler = TaskScheduler.FromCurrentSynchronizationContext();

            var task = new Task(action, TaskCreationOptions.LongRunning);
            task.ContinueWith(asyncPartTask => onException(asyncPartTask.Exception), CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, scheduler);
            task.ContinueWith(asyncPart =>
                                  {
                                      if (shouldComplete == null || shouldComplete())
                                          completition();
                                  },
                              CancellationToken.None, TaskContinuationOptions.OnlyOnRanToCompletion, scheduler);
            task.Start();
        }

        public static void ExecuteBlockingOperation<TIn>(Action<TIn> action, TIn parameter, Action<TIn> completition, Predicate<TIn> shouldComplete, Action<AggregateException, TIn> onException)
        {
            if (action == null)
                throw new ArgumentNullException("action");

            if (completition == null)
                throw new ArgumentNullException("completition");

            var scheduler = TaskScheduler.FromCurrentSynchronizationContext();

            var task = new Task(() => action(parameter), TaskCreationOptions.LongRunning);
            task.ContinueWith(asyncPartTask => onException(asyncPartTask.Exception, parameter), CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, scheduler);
            task.ContinueWith(asyncPart =>
                                  {
                                      if (shouldComplete == null || shouldComplete(parameter))
                                          completition(parameter);
                                  },
                              CancellationToken.None, TaskContinuationOptions.OnlyOnRanToCompletion, scheduler);
            task.Start();
        }

        public static void ExecuteBlockingOperation<TOut>(Func<TOut> func, Action<TOut> completition, Predicate<TOut> shoudComplete, Action<AggregateException> onException)
        {
            if (func == null)
                throw new ArgumentNullException("func");

            if (completition == null)
                throw new ArgumentNullException("completition");

            var scheduler = TaskScheduler.FromCurrentSynchronizationContext();

            var task = new Task<TOut>(func, TaskCreationOptions.LongRunning);
            task.ContinueWith(asyncPartTask => onException(asyncPartTask.Exception), CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, scheduler);
            task.ContinueWith(asyncPartTask =>
                                  {
                                      if (shoudComplete == null || shoudComplete(asyncPartTask.Result))
                                          completition(asyncPartTask.Result);
                                  },
                              CancellationToken.None, TaskContinuationOptions.OnlyOnRanToCompletion, scheduler);
            task.Start();
        }

        public static void ExecuteBlockingOperation<TIn, TOut>(Func<TIn, TOut> func, TIn parameter, Action<TOut, TIn> completition, Func<TOut, TIn, bool> shouldComplete, Action<AggregateException, TIn> onException)
        {
            if (func == null)
                throw new ArgumentNullException("func");

            if (completition == null)
                throw new ArgumentNullException("completition");

            var scheduler = TaskScheduler.FromCurrentSynchronizationContext();

            var task = new Task<TOut>(() => func(parameter), TaskCreationOptions.LongRunning);
            task.ContinueWith(asyncPartTask => onException(asyncPartTask.Exception, parameter), CancellationToken.None, TaskContinuationOptions.OnlyOnFaulted, scheduler); // on Exception
            task.ContinueWith(asyncPart =>
                                  {
                                      if (shouldComplete == null || shouldComplete(asyncPart.Result, parameter))
                                          completition(asyncPart.Result, parameter);
                                  },
                              CancellationToken.None, TaskContinuationOptions.OnlyOnRanToCompletion, scheduler);
            task.Start();
        }            
    }
Share:
14,459
Randel Ramirez
Author by

Randel Ramirez

{ C# | ASP.NET Core | Entity Framework Core | JavaScript | TypeScript | Web Applications | Mobile Applications | Xamarin | OOP } Get it to work. Make the code beautiful. Optimize.

Updated on June 12, 2022

Comments

  • Randel Ramirez
    Randel Ramirez almost 2 years

    I'm aware that with .net 4.5 there is the await, async keywords that allows for easy calling of async methods. I 'm currently studying how to make async calls in C# 4.0. An example I want to is to make an async call where the datagrid is being databind.

    If you could provide me some links I would really appreciate it.