无意当中看了一些博文,说有人想自己尝试实现基于异步操作的方法:

  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的更多相关文章

  1. [译] C# 5.0 中的 Async 和 Await (整理中...)

    C# 5.0 中的 Async 和 Await [博主]反骨仔 [本文]http://www.cnblogs.com/liqingwen/p/6069062.html 伴随着 .NET 4.5 和 V ...

  2. 探索c#之Async、Await剖析

    阅读目录: 基本介绍 基本原理剖析 内部实现剖析 重点注意的地方 总结 基本介绍 Async.Await是net4.x新增的异步编程方式,其目的是为了简化异步程序编写,和之前APM方式简单对比如下. ...

  3. Async和Await异步编程的原理

    1. 简介 从4.0版本开始.NET引入并行编程库,用户能够通过这个库快捷的开发并行计算和并行任务处理的程序.在4.5版本中.NET又引入了Async和Await两个新的关键字,在语言层面对并行编程给 ...

  4. 异步方法的意义何在,Async和await以及Task的爱恨情仇,还有多线程那一家子。

    前两天刚感受了下泛型接口的in和out,昨天就开始感受神奇的异步方法Async/await,当然顺路也看了眼多线程那几个.其实多线程异步相关的类单个用法和理解都不算困难,但是异步方法Async/awa ...

  5. 多线程之异步编程: 经典和最新的异步编程模型,async与await

    经典的异步编程模型(IAsyncResult) 最新的异步编程模型(async 和 await) 将 IAsyncInfo 转换成 Task 将 Task 转换成 IAsyncInfo 示例1.使用经 ...

  6. 浅谈async、await关键字 => 深谈async、await关键字

    前言 之前写过有关异步的文章,对这方面一直比较弱,感觉还是不太理解,于是会花点时间去好好学习这一块,我们由浅入深,文中若有叙述不稳妥之处,还请批评指正. 话题 (1)是不是将方法用async关键字标识 ...

  7. 使用 Async 和 Await 的异步编程(C# 和 Visual Basic)[msdn.microsoft.com]

    看到Microsoft官方一篇关于异步编程的文章,感觉挺好,不敢独享,分享给大家. 原文地址:https://msdn.microsoft.com/zh-cn/library/hh191443.asp ...

  8. 【转】【C#】C# 5.0 新特性——Async和Await使异步编程更简单

    一.引言 在之前的C#基础知识系列文章中只介绍了从C#1.0到C#4.0中主要的特性,然而.NET 4.5 的推出,对于C#又有了新特性的增加--就是C#5.0中async和await两个关键字,这两 ...

  9. C#基础系列——异步编程初探:async和await

    前言:前面有篇从应用层面上面介绍了下多线程的几种用法,有博友就说到了async, await等新语法.确实,没有异步的多线程是单调的.乏味的,async和await是出现在C#5.0之后,它的出现给了 ...

  10. C# Async与Await的使用

    这个是.NET 4.5的特性,所以要求最低.NET版本为4.5. 看很多朋友还是使用的Thread来使用异步多线程操作,基本上看不见有使用Async.Await进行异步编程的.各有所爱吧,其实都可以. ...

随机推荐

  1. springmvc初步配置

    导包/添加依赖: <dependency> <groupId>org.springframework</groupId> <artifactId>spr ...

  2. andorid avd 下 使用 fiddler 抓包

                Fiddler 设置:  

  3. 题目1022:游船出租(hash简单应用)

    问题来源 http://ac.jobdu.com/problem.php?pid=1022 问题描述 每次输入:船号(1~100) 键值(S或E) 发生时间(小时:分钟).当船号为0时,代表一天结束: ...

  4. sql 列集合

    STUFF((SELECT ','+CAST( TYZ_Bh  as varchar(10)) FROM #1 where 片区划分='江东' for xml path('')),1,1,'')

  5. pip安装本地文件

    I do a lot of development without an internet connection1, so being able to install packages into a ...

  6. weex 自定义Modul

    扩展iOS的功能 一. 新建 NSOjbect 子类, 并遵循协议<WXModuleProtocol> .h 代码 #import <Foundation/Foundation.h& ...

  7. 做point data的切面的时候的注意事项

    正确的顺序应该是: 先导入cell data,再转换为point data,再做切面.结果如下: 如果这里导入cell data以后先做了切面再转换为point data,结果就是这样的: 很明显中间 ...

  8. Python——单例设计模式

    单例设计模式: 让类创建的对象,在系统中只有唯一的实例, 使用python类内置的__new__()方法实现,__new__()方法在创建对象时会被自动调用,通过重写__new__()方法,使得无论用 ...

  9. 使用app-inspector查看元素,无法连接到手机,提示错误{ Error: Command failed ……forward tcp:9001 tcp:9001错误解决

    在学习使用app-inspector查看元素时,碰到一个问题.在cmd窗口执行命令app-inspector --port 5678 -u 85EABNFSU53R --verbose  ,连接不到手 ...

  10. rocketmq sql解析过滤

    activemq rocketmq kafka robbitmq 公司 apache alibaba LinkedIn Pivotal 编写语言 java Erlang 客户端支持 其他协议支持 mq ...