自己实现async和await
无意当中看了一些博文,说有人想自己尝试实现基于异步操作的方法:
1)直接使用Task(不说咯,这个是微软给我们的标准实现方法)。
2)必须继承INotifyCompletion接口,同时自己实现IsCompleted(必须)和Result(可选),GetResult(必须)和OnCompleted(必须)方法:
下面是一个具体的例子(自实现异步函数):
public interface IAwait<out T> : INotifyCompletion
{
bool IsCompleted { get; }
T Result { get; }
T GetResult();
}
public interface IAwaitable<out T>
{
IAwait<T> GetAwaiter();
} public class AwaitableFunc<T> : IAwaitable<T>
{
private Func<T> fun = null; public IAwait<T> GetAwaiter()
{
return new InnerAwaitableImplement(fun);
} public AwaitableFunc(Func<T> func)
{
fun = func;
} private class InnerAwaitableImplement : IAwait<T>
{
private Func<T> fun = null;
private bool isFinished=false;
private T result = default(T); public InnerAwaitableImplement(Func<T> func)
{
fun = func;
}
public bool IsCompleted
{
get
{
return isFinished;
}
} public T Result
{
get
{
return GetResult();
}
} public void OnCompleted(Action continuation)
{
ThreadPool.QueueUserWorkItem(obj =>
{
isFinished = true;
continuation();
}, null);
} public T GetResult()
{
result = fun();
return result;
}
}
}
GetResult和Result属性应该实现同步的方法(阻塞线程的),OnCompleted实现异步方法(必须新线程去处理)。这样的话,一旦主程序这样调用:
AwaitableFunc<int> afunc = new AwaitableFunc<int>(() =>
{
//模拟一个长时间的任务,注意这里如果用同步机器就死掉
Thread.Sleep();
return ;
}); var result = afunc.GetAwaiter(); result.OnCompleted(() =>
{
MessageBox.Show(result.GetResult().ToString());
});
你会发现,GetAwaiter方法会先被执行,判断IsCompleted是否为false,如果是false,先执行OnCompleted的方法(作为回调函数一样的性质)先保留,然后开辟新线程执行GetResult(),最后回调到OnCompleted执行回调函数。
你也可以这样调用:
private async void button1_Click(object sender, EventArgs e)
{
AwaitableFunc<int> afunc = new AwaitableFunc<int>(() =>
{
//模拟一个长时间的任务,注意这里如果用同步机器就死掉
Thread.Sleep();
return ;
}); var result = await afunc;
MessageBox.Show(result.ToString());
}
类似于第一个示例(这里就指出了await其实本质是一个回调函数,编译器自动把await下面的东西全部包含到里边去了,简单叙述原理,注意代码中红色标示部分的位置!)。
其实,GetResult并不是一定需要的,比如这个对int任意进行延时(不直接调用Task.Delay方法,自己写一个呗):
public class TimeDelay
{
private int _delayTime = ; public TimeDelay(int delayNumber)
{
_delayTime = delayNumber;
}
public InnerAwaitableImplement GetAwaiter()
{
return new InnerAwaitableImplement(_delayTime);
}
public class InnerAwaitableImplement:INotifyCompletion
{
private int _delayTime = ;
private bool isFinished=false; public InnerAwaitableImplement(int delayTime)
{
_delayTime = delayTime;
}
public bool IsCompleted
{
get
{
return isFinished;
}
}
public void OnCompleted(Action continuation)
{
ThreadPool.QueueUserWorkItem(obj =>
{
Thread.Sleep(_delayTime);
isFinished = true;
continuation();
}, null);
} public void GetResult()
{ }
}
}
这样使用:
private async void button1_Click(object sender, EventArgs e)
{
TimeDelay afunc = new TimeDelay();
await afunc;
MessageBox.Show("OK");
}
更简单地——扩展方法:
public class TimeDelay
{
private int _delayTime = ; public TimeDelay(int delayNumber)
{
_delayTime = delayNumber;
}
public InnerAwaitableImplement GetAwaiter()
{
return new InnerAwaitableImplement(_delayTime);
}
public class InnerAwaitableImplement:INotifyCompletion
{
private int _delayTime = ;
private bool isFinished=false; public InnerAwaitableImplement(int delayTime)
{
_delayTime = delayTime;
}
public bool IsCompleted
{
get
{
return isFinished;
}
}
public void OnCompleted(Action continuation)
{
ThreadPool.QueueUserWorkItem(obj =>
{
Thread.Sleep(_delayTime);
isFinished = true;
continuation();
}, null);
} public void GetResult()
{ }
}
} public static class IntExtend
{
public static TimeDelay.InnerAwaitableImplement GetAwaiter(this int delayTime)
{
TimeDelay td = new TimeDelay(delayTime);
return td.GetAwaiter();
}
}
这样调用:
private async void button1_Click(object sender, EventArgs e)
{await ;
MessageBox.Show("OK");
}
自己实现async和await的更多相关文章
- [译] C# 5.0 中的 Async 和 Await (整理中...)
C# 5.0 中的 Async 和 Await [博主]反骨仔 [本文]http://www.cnblogs.com/liqingwen/p/6069062.html 伴随着 .NET 4.5 和 V ...
- 探索c#之Async、Await剖析
阅读目录: 基本介绍 基本原理剖析 内部实现剖析 重点注意的地方 总结 基本介绍 Async.Await是net4.x新增的异步编程方式,其目的是为了简化异步程序编写,和之前APM方式简单对比如下. ...
- Async和Await异步编程的原理
1. 简介 从4.0版本开始.NET引入并行编程库,用户能够通过这个库快捷的开发并行计算和并行任务处理的程序.在4.5版本中.NET又引入了Async和Await两个新的关键字,在语言层面对并行编程给 ...
- 异步方法的意义何在,Async和await以及Task的爱恨情仇,还有多线程那一家子。
前两天刚感受了下泛型接口的in和out,昨天就开始感受神奇的异步方法Async/await,当然顺路也看了眼多线程那几个.其实多线程异步相关的类单个用法和理解都不算困难,但是异步方法Async/awa ...
- 多线程之异步编程: 经典和最新的异步编程模型,async与await
经典的异步编程模型(IAsyncResult) 最新的异步编程模型(async 和 await) 将 IAsyncInfo 转换成 Task 将 Task 转换成 IAsyncInfo 示例1.使用经 ...
- 浅谈async、await关键字 => 深谈async、await关键字
前言 之前写过有关异步的文章,对这方面一直比较弱,感觉还是不太理解,于是会花点时间去好好学习这一块,我们由浅入深,文中若有叙述不稳妥之处,还请批评指正. 话题 (1)是不是将方法用async关键字标识 ...
- 使用 Async 和 Await 的异步编程(C# 和 Visual Basic)[msdn.microsoft.com]
看到Microsoft官方一篇关于异步编程的文章,感觉挺好,不敢独享,分享给大家. 原文地址:https://msdn.microsoft.com/zh-cn/library/hh191443.asp ...
- 【转】【C#】C# 5.0 新特性——Async和Await使异步编程更简单
一.引言 在之前的C#基础知识系列文章中只介绍了从C#1.0到C#4.0中主要的特性,然而.NET 4.5 的推出,对于C#又有了新特性的增加--就是C#5.0中async和await两个关键字,这两 ...
- C#基础系列——异步编程初探:async和await
前言:前面有篇从应用层面上面介绍了下多线程的几种用法,有博友就说到了async, await等新语法.确实,没有异步的多线程是单调的.乏味的,async和await是出现在C#5.0之后,它的出现给了 ...
- C# Async与Await的使用
这个是.NET 4.5的特性,所以要求最低.NET版本为4.5. 看很多朋友还是使用的Thread来使用异步多线程操作,基本上看不见有使用Async.Await进行异步编程的.各有所爱吧,其实都可以. ...
随机推荐
- C++的可移植性和跨平台开发
概述 今天聊聊C++的可移植性问题.如果你平时使用C++进行开发,并且你对C++的可移植性问题不是非常清楚,那么我建议你看看这个系列.即使你目前没有跨平台开发的需要,了解可移植性方面的知识对你还是很有 ...
- “全栈2019”Java第六十七章:内部类、嵌套类详解
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- kafka-0.9
1)yum install java 2)curl -L -O http://mirrors.cnnic.cn/apache/kafka/0.9.0.0/kafka_2.10-0.9.0.0.tgz ...
- 题目1021:统计字符(hash简单应用)
问题来源 http://ac.jobdu.com/problem.php?pid=1021 问题描述 每次输入两个字符串,统计第一个字符串中的每个字符在第二个字符串中出现的次数. 问题分析 太明显了, ...
- 子元素margin-top为何会影响父元素?
子元素margin-top为何会影响父元素? 引用地址:https://blog.csdn.net/sinat_27088253/article/details/52954688 2016年10月28 ...
- SpringBoot入门(IDEA篇)(二)
一.SpringBoot启动的3种方式 第一种:借助IDE工具直接启动 run as 第二种:mvn命令启动 1:打开命令行,进入到项目目录中(我这里还是用上次建立的dog项目来操作)cd E:\Wo ...
- python函数参数类型及其顺序
根据inspect模块官文文档中关于函数参数类型的相关说明,python函数参数共有五种类型,按顺序分别为:POSITIONAL_ONLY.POSITIONAL_OR_KEYWORD.VAR_POSI ...
- es6精华
函数: ES6 允许为函数的参数设置默认值,即直接写在参数定义的后面.function log(x, y = 'World') { console.log(x, y); } log('Hello') ...
- mvc框架的学习步骤
1 搭建环境.有个感性认识 2 如何完成controller和viewer的映射 3 如何把值传给controller 4 controller如何把值传给view 5 异常处理 6 页面标签 7文件 ...
- OpenCV文本图像的旋转矫正
用户在使用Android手机拍摄过程中难免会出现文本图像存在旋转角度.这里采用霍夫变换.边缘检测等数字图像处理算法检测图像的旋转角度,并根据计算结果对输入图像进行旋转矫正. 首先定义一个结构元素,再通 ...