• 在 .NET 4.0 下编写扩展代码以支持 async 异步编程


      微软在C# 5中引入了async、await这两个异步编程的关键字,要使用这两个关键字需要你的IDE支持C#5.0语法,也就意味着你需要使用VS 2012版本以上IDE,或者在Vs2010卸载其编译器在重新安装,这并不是本文的重点,有需求的朋友点击这篇博客,现在都快接近2019年,VS2019 预览版都出来了,现在使用Vs2010确实有点更不上时代了,建议大家升级一下vs,其他问题可以自行百度谷歌解决下。

      async和await带给我们的方便是巨大的,从代码的简洁或者处理UI阻塞问题,都是很便捷的,而XP系统最高只支持.Net 4.0,也就意味着,假如你编写的Winform程序使用了这两个关键字将不能在XP中,而目前很多系统仍然使用着XP,因此很有必要在.Net 4 环境下使用 async异步编程。

      当我们创建一个.Net Framework 4.0的项目时候,在使用async 和 await的时候,代码如下:

            static void Main(string[] args)
            {
                Console.WriteLine("异步方法调用前:" + Thread.CurrentThread.ManagedThreadId.ToString());
                Test();
                Console.WriteLine("异步方法调用后:" + Thread.CurrentThread.ManagedThreadId.ToString());
                Console.Read();
            }
    
            public async static void Test()
            {
                Console.WriteLine("异步方法等待前:" + Thread.CurrentThread.ManagedThreadId.ToString());
                Console.WriteLine("开始等待:" + DateTime.Now.ToString());
                await Wait();
                Console.WriteLine("异步方法等待后:" + Thread.CurrentThread.ManagedThreadId.ToString());
                Console.WriteLine("结束等待:" + DateTime.Now.ToString());
            }
    
            public static Task Wait()
            {
                Action action = () =>
                {
                    Console.WriteLine("任务运行:" + Thread.CurrentThread.ManagedThreadId.ToString());
                    Thread.Sleep(2000);
                    Console.WriteLine("任务运行后:" + Thread.CurrentThread.ManagedThreadId.ToString());
                };
                Task task = null;
                //    task= TaskEx.Run(action);
                task = new Task(action);
                task.Start();
                return task;
            }

      这时候编译器会报一个错误 :CS1061 “Task”未包含“GetAwaiter”的定义,并且找不到可接受第一个“Task”类型参数的可访问扩展方法“GetAwaiter”(是否缺少 using 指令或程序集引用?) 

    而在.Net 4.5下,是不会报这样错的。在异步的环境下,正确执行的情况下,Test()方法会单独开辟一个线程,所以整个测试,会在Main方法中Test()方法是异步的,具体执行顺序如下:

      回归到.Net 4.0中报的错:“Task”未包含“GetAwaiter”的定义,并且找不到可接受第一个“Task”类型参数的可访问扩展方法“GetAwaiter”。由此我们推断.Net 4.5 跟 .Net 4.0在处理异步的区别是否在于多了Task的扩展方法GetAwaiter()。

    -------------------------------------------------------------------------------------华丽的过度线-------------------------------------------------------------------------------------------------------------------------

      由于涉及到 .net async 状态机,这东西目前不是目前我一下两下可以懂的,先将实现贴出来,以后学得深了在完善此随笔。

      新建一个类TaskEx,用如下的代码替换掉类里面的所有内容,会发现.Net 4.0下的报错会消失,也可以正常的使用异步async。有兴趣的同学可以在Github项目参考牛人写的代码

    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Runtime.CompilerServices;
    using System.Security;
    using System.Text;
    using System.Threading.Tasks;
    
    namespace System.Threading.Tasks
    {
        internal static class TaskEx
        {
            public static TaskAwaiter GetAwaiter(this Task task)
            {
                return new TaskAwaiter(task);
            }
    
            public static TaskAwaiter<T> GetAwaiter<T>(this Task<T> task)
            {
                return new TaskAwaiter<T>(task);
            }
            public static Task Run(Action action)
            {
                return Task.Factory.StartNew(action);
            }
        }
        internal struct TaskAwaiter : INotifyCompletion
        {
            readonly Task task;
            internal TaskAwaiter(Task task)
            {
                this.task = task;
            }
    
            internal static TaskScheduler TaskScheduler
            {
                get
                {
                    if (SynchronizationContext.Current == null)
                        return TaskScheduler.Default;
                    else
                        return TaskScheduler.FromCurrentSynchronizationContext();
                }
            }
    
            public bool IsCompleted
            {
                get { return task.IsCompleted; }
            }
    
            public void OnCompleted(Action continuation)
            {
                this.task.ContinueWith(
                    delegate (Task task) {
    
                        //Console.WriteLine("状态机回调函数信息:"+Thread.CurrentThread.ManagedThreadId);
                        //Console.WriteLine("	名称:"+continuation.Method.Name);
                        //Console.WriteLine("	所属类型:" + continuation.Method.DeclaringType.ToString());
                        continuation();
                    }, TaskAwaiter.TaskScheduler);
            }
    
            public void GetResult()
            {
                try
                {
                    task.Wait();
                }
                catch (AggregateException ex)
                {
                    throw ex.InnerExceptions[0];
                }
            }
        }
    
        internal struct TaskAwaiter<T> : INotifyCompletion
        {
            readonly Task<T> task;
    
            internal TaskAwaiter(Task<T> task)
            {
                this.task = task;
            }
    
            public bool IsCompleted
            {
                get { return task.IsCompleted; }
            }
    
            public void OnCompleted(Action continuation)
            {
                this.task.ContinueWith(
                    delegate (Task<T> task) {
                        continuation();
                    }, TaskAwaiter.TaskScheduler);
            }
    
            public T GetResult()
            {
                try
                {
                    return task.Result;
                }
                catch (AggregateException ex)
                {
                    throw ex.InnerExceptions[0];
                }
            }
        }
    }
    
    namespace System.Runtime.CompilerServices
    {
        internal interface INotifyCompletion
        {
            void OnCompleted(Action continuation);
        }
    
        internal interface ICriticalNotifyCompletion : INotifyCompletion
        {
            [SecurityCritical]
            void UnsafeOnCompleted(Action continuation);
        }
    
        internal interface IAsyncStateMachine
        {
            void MoveNext();
            void SetStateMachine(IAsyncStateMachine stateMachine);
        }
    
        internal struct AsyncVoidMethodBuilder
        {
            public static AsyncVoidMethodBuilder Create()
            {
                return new AsyncVoidMethodBuilder();
            }
    
            public void SetException(Exception exception)
            {
                throw exception;
            }
    
            public void SetResult()
            {
            }
    
            public void SetStateMachine(IAsyncStateMachine stateMachine)
            {
                // Should not get called as we don't implement the optimization that this method is used for.
                throw new NotImplementedException();
            }
    
            public void Start<TStateMachine>(ref TStateMachine stateMachine) where TStateMachine : IAsyncStateMachine
            {
                stateMachine.MoveNext();
            }
    
            public void AwaitOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine) where TAwaiter : INotifyCompletion where TStateMachine : IAsyncStateMachine
            {
                awaiter.OnCompleted(stateMachine.MoveNext);
            }
    
            public void AwaitUnsafeOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine) where TAwaiter : ICriticalNotifyCompletion where TStateMachine : IAsyncStateMachine
            {
                awaiter.OnCompleted(stateMachine.MoveNext);
            }
        }
    
        internal struct AsyncTaskMethodBuilder
        {
            TaskCompletionSource<object> tcs;
    
            public Task Task { get { return tcs.Task; } }
    
            public static AsyncTaskMethodBuilder Create()
            {
                AsyncTaskMethodBuilder b;
                b.tcs = new TaskCompletionSource<object>();
                return b;
            }
    
            public void Start<TStateMachine>(ref TStateMachine stateMachine) where TStateMachine : IAsyncStateMachine
            {
                stateMachine.MoveNext();
            }
    
            public void SetStateMachine(IAsyncStateMachine stateMachine)
            {
                // Should not get called as we don't implement the optimization that this method is used for.
                throw new NotImplementedException();
            }
    
            public void AwaitOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine) where TAwaiter : INotifyCompletion where TStateMachine : IAsyncStateMachine
            {
                awaiter.OnCompleted(stateMachine.MoveNext);
            }
    
            public void AwaitUnsafeOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine) where TAwaiter : ICriticalNotifyCompletion where TStateMachine : IAsyncStateMachine
            {
                awaiter.OnCompleted(stateMachine.MoveNext);
            }
    
            public void SetResult()
            {
                tcs.SetResult(null);
            }
    
            public void SetException(Exception exception)
            {
                tcs.SetException(exception);
            }
        }
    
        internal struct AsyncTaskMethodBuilder<T>
        {
            TaskCompletionSource<T> tcs;
    
            public Task<T> Task { get { return tcs.Task; } }
    
            public static AsyncTaskMethodBuilder<T> Create()
            {
                AsyncTaskMethodBuilder<T> b;
                b.tcs = new TaskCompletionSource<T>();
                return b;
            }
    
            public void Start<TStateMachine>(ref TStateMachine stateMachine) where TStateMachine : IAsyncStateMachine
            {
                stateMachine.MoveNext();
            }
    
            public void SetStateMachine(IAsyncStateMachine stateMachine)
            {
                // Should not get called as we don't implement the optimization that this method is used for.
                throw new NotImplementedException();
            }
    
            public void AwaitOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine) where TAwaiter : INotifyCompletion where TStateMachine : IAsyncStateMachine
            {
                awaiter.OnCompleted(stateMachine.MoveNext);
            }
    
            public void AwaitUnsafeOnCompleted<TAwaiter, TStateMachine>(ref TAwaiter awaiter, ref TStateMachine stateMachine) where TAwaiter : ICriticalNotifyCompletion where TStateMachine : IAsyncStateMachine
            {
                AwaitOnCompleted(ref awaiter, ref stateMachine);
            }
    
            public void SetResult(T result)
            {
                tcs.SetResult(result);
            }
    
            public void SetException(Exception exception)
            {
                tcs.SetException(exception);
            }
        }
    }
  • 相关阅读:
    安卓学习39
    安卓学习38
    Python+Selenium学习--打印当前页面的title及url
    Python+Selenium学习--访问连接
    Python+Selenium学习--浏览器设置
    Python+Selenium学习--启动及关闭浏览器
    Go语言学习笔记(十八)之文件读写
    Go语言学习笔记(十七)之命令行参数
    Go语言学习笔记(十六)之格式化输入输出
    Go语言学习笔记(十五)之异常处理
  • 原文地址:https://www.cnblogs.com/fode/p/10101456.html
Copyright © 2020-2023  润新知