tnblog
首页
视频
资源
登录

.net core Task的学习理解(下)

7704人阅读 2021/8/20 13:31 总访问:3660642 评论:0 收藏:0 手机
分类: .net后台框架

.netcore

.net core Task的学习理解(下)

Task.GetResult与死锁


通过下面的代码,可能会造成死锁的情况。

  1. static async Task Main(string[] args)
  2. {
  3. Console.WriteLine("Test:Start");
  4. DoSthAsync().GetAwaiter().GetResult();
  5. Console.WriteLine("Test:END");
  6. Console.ReadLine();
  7. }
  8. static async Task DoSthAsync()
  9. {
  10. Console.WriteLine("DoSthAsync: START");
  11. await Task.Delay(100);
  12. Console.WriteLine("DoSthAsync: END");
  13. }


但是通过如下代码就可以完成运行。这将与同步上下文有关,也就是SynchronizationContext

  1. static async Task Main(string[] args)
  2. {
  3. Console.WriteLine("Test:Start");
  4. DoSthAsync().GetAwaiter().GetResult();
  5. Console.WriteLine("Test:END");
  6. Console.ReadLine();
  7. }
  8. static async Task DoSthAsync()
  9. {
  10. Console.WriteLine("DoSthAsync: START");
  11. await Task.Delay(100).ConfigureAwait(continueOnCapturedContext:false);
  12. Console.WriteLine("DoSthAsync: END");
  13. }

SynchronizationContext与Task回调处理


Task不仅可以让TaskScheduler调用,还可以使用SynchronizationContext。
TaskAwaiter在实际注册回调的时候会对回调进行包装,决定了回调会在哪里执行。
当SynchronizationContext.Current不为为null时将会到SynchronizationContext上执行回调,为空是才会去判断TaskScheduler。(具体示意图如下)

自定义SynchronizationContext

  1. class MaxConcurrencySynchronizationContext: SynchronizationContext
  2. {
  3. /// <summary>
  4. /// 创建一个信号量
  5. /// </summary>
  6. private readonly SemaphoreSlim _semaphore;
  7. public MaxConcurrencySynchronizationContext(int maxConcurrencyLevel) =>
  8. _semaphore = new SemaphoreSlim(maxConcurrencyLevel);
  9. public override void Post(SendOrPostCallback d, object state)
  10. {
  11. _semaphore.Wait();
  12. try
  13. {
  14. Console.WriteLine("MaxConcurrencySynchronizationContext.ThreadId:{0}",Thread.CurrentThread.ManagedThreadId);
  15. d(state);
  16. }
  17. finally
  18. {
  19. _semaphore.Release();
  20. }
  21. }
  22. public override void Send(SendOrPostCallback d, object state) => throw new NotImplementedException();
  23. }


添加测试代码

  1. static void Main(string[] args)
  2. {
  3. Task.Run(() =>
  4. {
  5. var mcs = new MaxConcurrencySynchronizationContext(1);
  6. // 设置为我们自定义的同步上下文
  7. SynchronizationContext.SetSynchronizationContext(mcs);
  8. mcs.Post(_ => { Test(); }, null);
  9. });
  10. Console.ReadLine();
  11. }
  12. static void Test()
  13. {
  14. Console.WriteLine("Test:Start");
  15. DoSthAsync().GetAwaiter().GetResult();
  16. Console.WriteLine("Test:END");
  17. }
  18. static async Task DoSthAsync()
  19. {
  20. Console.WriteLine("DoSthAsync: START");
  21. await Task.Delay(100);
  22. Console.WriteLine("DoSthAsync: END");
  23. }


由于最大等待数量为1,所以当我们第二次在Task.Delay的时候就会被阻塞起。


而之前我们看到的.ConfigureAwait(continueOnCapturedContext:false);,这段代码的意思就是:不管你有没有同步上下文我都会到TaskScheduler上去执行。

  1. static void Main(string[] args)
  2. {
  3. Task.Run(() =>
  4. {
  5. var mcs = new MaxConcurrencySynchronizationContext(1);
  6. // 设置为我们自定义的同步上下文
  7. SynchronizationContext.SetSynchronizationContext(mcs);
  8. mcs.Post(_ => { Test(); }, null);
  9. });
  10. Console.ReadLine();
  11. }
  12. static void Test()
  13. {
  14. Console.WriteLine("Test:Start");
  15. DoSthAsync().GetAwaiter().GetResult();
  16. Console.WriteLine("Test:END");
  17. }
  18. static async Task DoSthAsync()
  19. {
  20. Console.WriteLine("DoSthAsync: START");
  21. await Task.Delay(100).ConfigureAwait(false);
  22. Console.WriteLine("DoSthAsync: END");
  23. }

ValueTask


ValueTask与Task类似,但它是值类型的。ValueTask<TResult>提供了一个AsTask方法,可根据需要获取的常规task(例如用作Task.WhenAll或Task.WhenAny调用的某个元素),不过多数情况下只是安装普通task那样调用await操作。

ValueTask<TResult>相比于Task<TResult>优势何在?其实体现在堆内存分配和垃圾回收上。Task<TResult>是一个类,虽然有时异步基础架构会复用已创建的Task<TResult>对象。但多数async方法需要创建新的Task<TResult>对象。一般情况下,.NET创建对象的性能消耗不足为虑,但如果频繁创建对象或者遇到性能敏感的代码,就需要尽量避免创建新对象。

如果在async方法中对某个尚未完成的操作使用await,那么创建对象是不可避免的,虽然此时方法会立即返回,但它需要安排一个延续。当操作完成时,该延续负责执行async方法中的其他语句。大部分async方法中的操作不会在await前执行完成。此时使用使用ValueTask<TResult>没有任何优势,可能还会导致性能下降。

有时task在await之前便已完成,这个时候就能体现出ValueTask<TResult>的强大了

  1. public sealed class ByteStream : IDisposable
  2. {
  3. private readonly Stream stream;
  4. private readonly byte[] buffer;
  5. private int position;
  6. private int bufferedBytes;
  7. public ByteStream(Stream stream)
  8. {
  9. this.stream = stream;
  10. // 8KB 缓冲区的大小,根本不需要await操作
  11. buffer = new byte[1024 * 8];
  12. }
  13. public async ValueTask<byte?> ReadByteAsync()
  14. {
  15. if (position == bufferedBytes)
  16. {
  17. position = 0;
  18. // 读取数据且不需要同步上下文执行
  19. bufferedBytes = await stream.ReadAsync(buffer, 0, buffer.Length).ConfigureAwait(false);
  20. // 读到末尾了
  21. if (bufferedBytes == 0)
  22. {
  23. return null;
  24. }
  25. }
  26. // 返回缓冲区的下一个字节
  27. return buffer[position++];
  28. }
  29. public void Dispose()
  30. {
  31. stream.Dispose();
  32. }
  33. }


欢迎加群讨论技术,1群:677373950(满了,可以加,但通过不了),2群:656732739

评价

.net core Task的学习理解(上)

.net core Task的学习理解(上)[TOC] Thread 线程的生命周期 计算机系统在某一个时刻,当只有一个CPU工作时,它只执行一个...

net core 使用 EF Code First

下面这些内容很老了看这篇:https://www.tnblog.net/aojiancc2/article/details/5365 项目使用多层,把数据库访问...

.net mvc分部页,.net core分部页

.net分部页的三种方式第一种:@Html.Partial(&quot;_分部页&quot;)第二种:@{ Html.RenderPartial(&quot;分部页&quot;);}...

StackExchange.Redis操作redis(net core支持)

官方git开源地址https://github.com/StackExchange/StackExchange.Redis官方文档在docs里边都是官方的文档通过nuget命令下...

.net core 使用session

tip:net core 2.2后可以直接启用session了,不用在自己添加一次session依赖,本身就添加了使用nuget添加引用Microsoft.AspN...

通俗易懂,什么是.net?什么是.net Framework?什么是.net core?

朋友圈@蓝羽 看到一篇文章写的太详细太通俗了,搬过来细细看完,保证你对.NET有个新的认识理解原文地址:https://www.cnblo...

asp.net core2.0 依赖注入 AddTransient与AddScoped的区别

asp.net core主要提供了三种依赖注入的方式其中AddTransient与AddSingleton比较好区别AddTransient瞬时模式:每次都获取一...

.net core 使用 Kestrel

Kestrel介绍 Kestrel是一个基于libuv的跨平台web服务器 在.net core项目中就可以不一定要发布在iis下面了Kestrel体验可以使...

net core中使用cookie

net core中可以使用传统的cookie也可以使用加密的cookieNET CORE中使用传统cookie设置:HttpContext.Response.Cookies.Appe...

net core项目结构简单分析

一:wwwrootwwwroot用于存放网站的静态资源,例如css,js,图片与相关的前端插件等lib主要是第三方的插件,例如微软默认引用...

net core使用EF之DB First

一.新建一个.net core的MVC项目新建好项目后,不能像以前一样直接在新建项中添加ef了,需要用命令在添加ef的依赖二.使用Nug...

.net core使用requestresponse下载文件下载excel等

使用request获取内容net core中request没有直接的索引方法,需要点里边的Query,或者formstringbase64=Request.Form[&quot;f...

iframe自适应高度与配合net core使用

去掉iframe边框frameborder=&quot;0&quot;去掉滚动条scrolling=&quot;no&quot;iframe 自适应高度如果内容是固定的,那么就...

net core启动报错Unable to configure HTTPS endpoint. No server certificate was specified

这是因为net core2.1默认使用的https,如果使用Kestrel web服务器的话没有安装证书就会报这个错其实仔细看他的错误提示,其...

net core中使用url编码与解码操作

net core中暂时还没有以前asp.net与mvc中的server对象。获取url的编码与解码操作不能使用以前的server对象来获取。使用的是...
这一世以无限游戏为使命!
排名
2
文章
657
粉丝
44
评论
93
docker中Sware集群与service
尘叶心繁 : 想学呀!我教你呀
一个bug让程序员走上法庭 索赔金额达400亿日元
叼着奶瓶逛酒吧 : 所以说做程序员也要懂点法律知识
.net core 塑形资源
剑轩 : 收藏收藏
映射AutoMapper
剑轩 : 好是好,这个对效率影响大不大哇,效率高不高
ASP.NET Core 服务注册生命周期
剑轩 : http://www.tnblog.net/aojiancc2/article/details/167
ICP备案 :渝ICP备18016597号-1
网站信息:2018-2025TNBLOG.NET
技术交流:群号656732739
联系我们:contact@tnblog.net
公网安备:50010702506256
欢迎加群交流技术