当前位置:首页 > 科技  > 软件

.Net8顶级性能优化:类型转换

来源: 责编: 时间:2023-11-06 17:19:51 418观看
导读1.前言.Net8通过各种骚操,把性能提升到了前所未有的高度。超越以往任何版本,也涵盖了后续版本,比如.NET9或许可能没有如此大的性能优化了。本篇来看下它其中的一个优化:类型转换的优化效果。2.示例通过类型检查的优化,优

1.前言

.Net8通过各种骚操,把性能提升到了前所未有的高度。超越以往任何版本,也涵盖了后续版本,比如.NET9或许可能没有如此大的性能优化了。本篇来看下它其中的一个优化:类型转换的优化效果。yjF28资讯网——每日最新资讯28at.com

2.示例

通过类型检查的优化,优化掉某些情况下类型转换的时候JIT类型检查的函数。下面的代码是类型检查的典型应用。yjF28资讯网——每日最新资讯28at.com

[HideColumns("Error", "StdDev", "Median", "RatioSD")][DisassemblyDiagnoser(maxDepth: 0)]public class Tests{  private readonly string[] _strings = new string[1];  [Benchmark]  public string Get1() => _strings[0];  [Benchmark]  public string Get2() => Volatile.Read(ref _strings[0]);}public partial class Program{   static void Main(string[] args)   {     BenchmarkSwitcher.FromAssembly(typeof(Tests).Assembly).Run(args);   }}

我们看到_strings是个私有数组,Get1函数中获取_strings数组的第一个值。所以它是直接用ldelem.ref IL执行即可yjF28资讯网——每日最新资讯28at.com

ldelem.ref

但是Get2里面对数组元素进行了引用,所以Roslyn的指令是:yjF28资讯网——每日最新资讯28at.com

ldelema [System.Runtime]System.String

如果ref类型的变量,被赋值为不同于这个变量的类型则会违反类型安全性。通常情况下ldelema需要进行类型检查,也就是用JIT辅助函数CORINFO_HELP_LDELEMA_REF来进行检查,以确保不会违反类型安全性。yjF28资讯网——每日最新资讯28at.com

这个安全性的检查会极大损耗性能,.NET8的JIT进行了一个优化,思路是如果是sealed关键字标记的类型,就不会进行安全性检查,这样就会提高性能。为什么sealed不会呢?yjF28资讯网——每日最新资讯28at.com

这其实是利用了它的一个特性,就是不会被继承。不会被继承,就不会被子类的类型所困扰,只有string一个类型,自然不会用以进行类型检查了。yjF28资讯网——每日最新资讯28at.com

yjF28资讯网——每日最新资讯28at.com

这是第一点优化,下面看下。yjF28资讯网——每日最新资讯28at.com

3.第一阶优化

优化了类型安全检查,缩短了编译时间,提高了性能。来看下.Net7和.NET8的生成Get2函数的的不同点yjF28资讯网——每日最新资讯28at.com

.Net7:yjF28资讯网——每日最新资讯28at.com

Tests.Get2()       sub       rsp,28       mov       rcx,[rcx+8]       xor       edx,edx       mov       r8,offset MT_System.String       call      CORINFO_HELP_LDELEMA_REF       mov       rax,[rax]       add       rsp,28       ret; Total bytes of code 33

.Net7它这里有一个CORINFO_HELP_LDELEMA_REF进行安全性检查。yjF28资讯网——每日最新资讯28at.com

.Net8:yjF28资讯网——每日最新资讯28at.com

; Tests.Get2()       sub       rsp,28       mov       rax,[rcx+8]       cmp       dword ptr [rax+8],0       jbe       short M00_L00       mov       rax,[rax+10]       add       rsp,28       retM00_L00:       call      CORINFO_HELP_RNGCHKFAIL       int       3; Total bytes of code 29

.Net8里它没有了CORINFO_HELP_LDELEMA_REFyjF28资讯网——每日最新资讯28at.com

因为string类型是sealed,它的原型如下:yjF28资讯网——每日最新资讯28at.com

public sealed class String : IEnumerable<char>, IEnumerable, ICloneable, IComparable, IComparable<String?>, IConvertible, IEquatable<String?>{  //这里代码省略}

JIT会判断类型是否是sealed标志,如果是则不进行安全性检查优化。yjF28资讯网——每日最新资讯28at.com

虽然.Net8去掉了CORINFO_HELP_LDELEMA_REF,yjF28资讯网——每日最新资讯28at.com

但是多了范围的检查CORINFO_HELP_RNGCHKFAIL,那它这个性能如何呢?yjF28资讯网——每日最新资讯28at.com

我们来测试下:yjF28资讯网——每日最新资讯28at.com

dotnet run -c Release -f net7.0 --filter "*" --runtimes net7.0 net8.0

结果是:yjF28资讯网——每日最新资讯28at.com

Method
yjF28资讯网——每日最新资讯28at.com

RuntimeyjF28资讯网——每日最新资讯28at.com

Mean
yjF28资讯网——每日最新资讯28at.com

Ratio
yjF28资讯网——每日最新资讯28at.com

Code Size
yjF28资讯网——每日最新资讯28at.com

Get2yjF28资讯网——每日最新资讯28at.com

.NET 7.0yjF28资讯网——每日最新资讯28at.com

1.0537 nsyjF28资讯网——每日最新资讯28at.com

1.00yjF28资讯网——每日最新资讯28at.com

33 ByjF28资讯网——每日最新资讯28at.com

Get2yjF28资讯网——每日最新资讯28at.com

.NET 8.0yjF28资讯网——每日最新资讯28at.com

0.2423 nsyjF28资讯网——每日最新资讯28at.com

0.23yjF28资讯网——每日最新资讯28at.com

29 ByjF28资讯网——每日最新资讯28at.com

我们看到同样代码,.Net8里面比.Net7的性能提升了5倍之多。yjF28资讯网——每日最新资讯28at.com

4.第二阶优化

承接上面,上面sealed去掉了类型检查。yjF28资讯网——每日最新资讯28at.com

然后在类型转换的时候,一般的类型转换JIT使用的是CastHelpers.ChkCastAny来进行。yjF28资讯网——每日最新资讯28at.com

但是.Net8里面内联了一个方法yjF28资讯网——每日最新资讯28at.com

用以缩短CastHelpers.ChkCastAny的编译时间,提高编译的时间和程序的性能。yjF28资讯网——每日最新资讯28at.com

using BenchmarkDotNet.Attributes;using BenchmarkDotNet.Running;using System.Runtime.CompilerServices;BenchmarkSwitcher.FromAssembly(typeof(Tests).Assembly).Run(args);[HideColumns("Error", "StdDev", "Median", "RatioSD")]public class Tests{    private readonly object _o = "hello";    [Benchmark]    public string GetString() => Cast<string>(_o);    [MethodImpl(MethodImplOptions.NoInlining)]    public T Cast<T>(object o) => (T)o;}

同样的yjF28资讯网——每日最新资讯28at.com

dotnet run -c Release -f net7.0 --filter "*" --runtimes net7.0 net8.0

结果如下:yjF28资讯网——每日最新资讯28at.com

Method
yjF28资讯网——每日最新资讯28at.com

Runtime
yjF28资讯网——每日最新资讯28at.com

Mean
yjF28资讯网——每日最新资讯28at.com

Ratio
yjF28资讯网——每日最新资讯28at.com

GetStringyjF28资讯网——每日最新资讯28at.com

.NET 7.0yjF28资讯网——每日最新资讯28at.com

3.018 nsyjF28资讯网——每日最新资讯28at.com

1.00yjF28资讯网——每日最新资讯28at.com

GetStringyjF28资讯网——每日最新资讯28at.com

.NET 8.0yjF28资讯网——每日最新资讯28at.com

1.198 nsyjF28资讯网——每日最新资讯28at.com

0.40yjF28资讯网——每日最新资讯28at.com

.Net8是三倍于.Net7的运行速度。去掉类型检查+类型转换的内联,大幅度的提升效率,可见.Net8的性能优化确实不容小觑。yjF28资讯网——每日最新资讯28at.com

参考如下:yjF28资讯网——每日最新资讯28at.com

https://devblogs.microsoft.com/dotnet/performance-improvements-in-net-8/

最后推荐下个人的CLR/JIT交流圈,里面有多篇个人编写的高质量的原创栏目和文章。学习心得,项目经验等。带你进入.Net核心技术阶层,脱离curd工程师范畴。yjF28资讯网——每日最新资讯28at.com

yjF28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-17278-0.html.Net8顶级性能优化:类型转换

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: Go 与数据可视化:使用 Gonum 和 Plot 库探索数据之美

下一篇: 携程后台低代码平台的探究与实践

标签:
  • 热门焦点
  • 服务存储设计模式:Cache-Aside模式

    Cache-Aside模式一种常用的缓存方式,通常是把数据从主存储加载到KV缓存中,加速后续的访问。在存在重复度的场景,Cache-Aside可以提升服务性能,降低底层存储的压力,缺点是缓存和底
  • 一年经验在二线城市面试后端的经验分享

    忠告这篇文章只适合2年内工作经验、甚至没有工作经验的朋友阅读。如果你是2年以上工作经验,请果断划走,对你没啥帮助~主人公这篇文章内容来自 「升职加薪」星球星友 的投稿,坐
  • 从 Pulsar Client 的原理到它的监控面板

    背景前段时间业务团队偶尔会碰到一些 Pulsar 使用的问题,比如消息阻塞不消费了、生产者消息发送缓慢等各种问题。虽然我们有个监控页面可以根据 topic 维度查看他的发送状态,
  • Flowable工作流引擎的科普与实践

    一.引言当我们在日常工作和业务中需要进行各种审批流程时,可能会面临一系列技术和业务上的挑战。手动处理这些审批流程可能会导致开发成本的增加以及业务复杂度的上升。在这
  • 使用LLM插件从命令行访问Llama 2

    最近的一个大新闻是Meta AI推出了新的开源授权的大型语言模型Llama 2。这是一项非常重要的进展:Llama 2可免费用于研究和商业用途。(几小时前,swyy发现它已从LLaMA 2更名为Lla
  • 从零到英雄:高并发与性能优化的神奇之旅

    作者 | 波哥审校 | 重楼作为公司的架构师或者程序员,你是否曾经为公司的系统在面对高并发和性能瓶颈时感到手足无措或者焦头烂额呢?笔者在出道那会为此是吃尽了苦头的,不过也得
  • 余承东:AI大模型技术的发展将会带来下一代智能终端操作系统的智慧体验

    8月4日消息,2023年华为开发者大会(HDC.Together)今天正式开幕,华为发布HarmonyOS 4、全新升级的鸿蒙开发套件、HarmonyOS Next开发者预览版本等一系列
  • Counterpoint :OPPO双旗舰战略全面落地 高端产品销量增长22%

    2023年6月30日,全球行业分析机构Counterpoint Research发布的《中国智能手机高端市场白皮书》显示,中国智能手机品牌正在寻求高质量发展,中国高端智能
  • 朋友圈可以修改可见范围了 苹果用户可率先体验

    近日,iOS用户迎来微信8.0.27正式版更新,除了可更换二维码背景外,还新增了多项实用功能。在新版微信中,朋友圈终于可以修改可见范围,简单来说就是已发布的朋友圈
Top