0. 前言

首先立马解释一波为啥会有这样一篇伪标题的Demo随笔呢?

不是本人有知识误区,或者要误人子弟

因为大家都知道emit写出来的都是同步方法,不可能await,至少现在这么多年来没有提供对应的功能

这是之前某天在微信群看见讨论怎么emit一个异步方法并包装异步结构,简单几句文字也未能清晰的表达

所以趁着元旦节放假有点时间,

简单列举三种我知道方式去达到这样的效果

三种方法都是绕过emit直接书写emit代码,而是将对应逻辑转到其他方法中,最后emit调用方法达到效果

Demo 说明

原始方法是个延迟2秒之后返回55的方法:

        public static async Task<int> GetV()
{
await Task.Delay(2000);
return 55;
}

现在我们需要把 55 的结果加 6 ,让最终的结果变为 61

我们的测试方法是这样,会输出一些简单的时间,帮助我们了解执行顺序和异步情况

        private static async Task Test(MethodInfo method, MethodInfo awaitMehtod)
{
var caller = CreateCaller(method, awaitMehtod);
Console.WriteLine($"Start {awaitMehtod.Name} at: {DateTime.Now}.");
var task = caller();
Console.WriteLine($"Call done at: {DateTime.Now}.");
var number = await task;
Console.WriteLine($"Hello {number} at: {DateTime.Now}.");
Console.WriteLine($"End at: {DateTime.Now}.");
Console.WriteLine();
}

1. ContinueWith

        public static Func<Task<int>> CreateCaller(MethodInfo method, MethodInfo awaitMehtod)
{
var m = new DynamicMethod(Guid.NewGuid().ToString("N"), typeof(Task<int>), Type.EmptyTypes);
var il = m.GetILGenerator();
il.Emit(OpCodes.Call, method);
il.Emit(OpCodes.Call, typeof(Program).GetMethod(nameof(Program.AddSixUseContinueWith))); // 这里是差异点
il.Emit(OpCodes.Ret); return m.CreateDelegate(typeof(Func<Task<int>>)) as Func<Task<int>>;
} public static Task<int> AddSixUseContinueWith(Task<int> task)
{
return task.ContinueWith(i =>
{
Console.WriteLine($"AddSixUseContinueWith is: {DateTime.Now}.");
return i.Result + 6;
});
}

测试结果:

Start AddSixUseContinueWith at: 2021/1/2 13:34:55.
Call done at: 2021/1/2 13:34:55.
AddSixUseContinueWith is: 2021/1/2 13:34:57.
Hello 61 at: 2021/1/2 13:34:57.
End at: 2021/1/2 13:34:57.

优点

还是真正的异步

缺点

成本比较大,毕竟这样没有了状态机等等优化,(成本在 ns 级别哦,不是大家想的 ms哦)

2. GetAwaiter().GetResult()

        public static Func<Task<int>> CreateCaller(MethodInfo method, MethodInfo awaitMehtod)
{
var m = new DynamicMethod(Guid.NewGuid().ToString("N"), typeof(Task<int>), Type.EmptyTypes);
var il = m.GetILGenerator();
il.Emit(OpCodes.Call, method);
il.Emit(OpCodes.Call, typeof(Program).GetMethod(nameof(Program.AddSixUseAwaiter))); // 这里是差异点
il.Emit(OpCodes.Ret); return m.CreateDelegate(typeof(Func<Task<int>>)) as Func<Task<int>>;
} public static Task<int> AddSixUseAwaiter(Task<int> task)
{
var r = task.ConfigureAwait(false).GetAwaiter().GetResult() + 6;
Console.WriteLine($"AddSixUseAwaiter is: {DateTime.Now}.");
return Task.FromResult(r);
}

测试结果:

Start AddSixUseAwaiter at: 2021/1/2 13:34:57.
AddSixUseAwaiter is: 2021/1/2 13:34:59.
Call done at: 2021/1/2 13:34:59.
Hello 61 at: 2021/1/2 13:34:59.
End at: 2021/1/2 13:34:59.

优点

执行时间上消耗很小

缺点

当然这样 异步都变成了同步,所以可能会在某些情况下我们操作不当的代码从而导致失去异步方法的优势

3. async/await

        public static Func<Task<int>> CreateCaller(MethodInfo method, MethodInfo awaitMehtod)
{
var m = new DynamicMethod(Guid.NewGuid().ToString("N"), typeof(Task<int>), Type.EmptyTypes);
var il = m.GetILGenerator();
il.Emit(OpCodes.Call, method);
il.Emit(OpCodes.Call, typeof(Program).GetMethod(nameof(Program.AddSixUseAsyncAwait))); // 这里是差异点
il.Emit(OpCodes.Ret); return m.CreateDelegate(typeof(Func<Task<int>>)) as Func<Task<int>>;
} public static async Task<int> AddSixUseAsyncAwait(Task<int> task)
{
var r = await task;
Console.WriteLine($"AddSixUseAsyncAwait is: {DateTime.Now}.");
return r + 6;
}

测试结果:

Start AddSixUseAsyncAwait at: 2021/1/2 13:34:59.
Call done at: 2021/1/2 13:34:59.
AddSixUseAsyncAwait is: 2021/1/2 13:35:01.
Hello 61 at: 2021/1/2 13:35:01.
End at: 2021/1/2 13:35:01.

优点

async / await 本身的优势都没有损失

缺点

原本想在 emit 中 对result的处理逻辑 必须迁移到 async / await 方法中,emit代码必须好好设计

完整Demo放在

https://github.com/fs7744/grocery/blob/main/csharp/emit_await/EmitAwaitDemo/Program.cs

分享不易,如果能给予一点动力,不胜感激:关注一下本人的开源项目: Norns.Urd

最新文章

  1. windows部署React-Native的开发环境实践(技术细节)
  2. swoole
  3. IOS之Core Foundation框架和Cocoa Foundation框架的区别
  4. java 垃圾回收(堆内存)、以及栈内存的释放
  5. Python3基础 map 与 lambda表达式配合 将指定系列元素乘2
  6. hdu1166树状数组
  7. [翻译]成为顶尖程序员应当学什么?Python、C还是Ruby?
  8. stylus选中hover元素的兄弟元素下的子元素
  9. 正则表达式和re模块
  10. ModelState 错误信息输出
  11. classmethod 和 staticmethod
  12. Docker for windows WIN版本,主板特性问题
  13. Css3动画属性总汇
  14. CSS/让一个盒子消失的5中方法
  15. Codeforces Round #284 (Div. 1) C. Array and Operations 二分图匹配
  16. 【计算机网络基础】TCP/IP、HTTP、Socket的概念
  17. Linux定时运行程序脚本
  18. Scala入门系列(二):条件控制与循环
  19. sort函数(cmp)、map用法---------------Tju_Oj_2312Help Me with the Game
  20. SQLSERVER2014集群实战——DNS的坑

热门文章

  1. 转:HTTP协议简介与在python中的使用详解
  2. 第15.32节 PyQt(Python+Qt)入门学习:containers容器类部件QToolBox工具箱介绍及使用案例
  3. 第11.21节 Python 中正则表达式的其他扩展功能
  4. PyQt(Python+Qt)学习随笔:QTableView的wordWrap属性
  5. ATT&amp;CK 实战 - 红日安全 vulnstack (二) 环境部署(劝退水文)
  6. 【面试题】GC Root都有哪些?
  7. x++ 和 ++x的区别
  8. Spring framework核心
  9. rsync+inotify-tools实时备份脚本
  10. 微信小程序下拉加载下一页