C# 泛型编译特性对性能的影响
C#作为一种强类型语言,具有丰富的泛型支持,允许开发者编写可以应对不同数据类型的通用代码。然而,在泛型编译时,针对结构和类作为泛型参数时,会对性能产生不同的影响。
泛型编译行为
在C#中,泛型编译行为取决于泛型参数的类型。具体而言,当泛型参数是结构(Struct)时,编译器会针对每个具体的结构类型生成特定的实现。而当泛型参数是类(Class)时,编译器则可能生成更通用的实现。
结构 vs 类
结构(Struct)
结构是值类型,它们存储在栈上,具有较小的内存开销。当泛型参数是结构时,编译器会针对每个具体的结构类型生成专门的实现,这可能导致更高的性能。因为每个结构类型都有自己的实现,避免了装箱和拆箱的开销,同时优化了内存分配和访问。
类(Class)
类是引用类型,存储在堆上,需要通过引用进行访问。当泛型参数是类时,编译器可能生成更通用的实现。这可能导致较低的性能,因为通用实现需要进行动态调度和引用类型的操作,增加了一些开销。
测试性能差异
针对不同的泛型参数进行性能测试是一种有效的方法,以观察结构和类对泛型编译特性的影响。在测试中,可能会发现对结构类型的泛型参数,其性能可能更高,而对类类型的泛型参数,其性能可能略低。
using System.Diagnostics;
namespace ConsoleApp1 {
internal interface IValueGetter {
int GetValue(int index);
}
internal class MyTestClass<T> where T : IValueGetter {
private readonly T _valueGetter;
public MyTestClass(T valueGetter) {
_valueGetter = valueGetter;
}
public void Run() {
long r = 0L;
for (int i = 0; i < int.MaxValue; i++) {
r += _valueGetter.GetValue(i);
}
}
}
internal struct StructValueGetter : IValueGetter {
public readonly int GetValue(int index) {
return index + 3;
}
}
internal struct StructValueGetter2(int someField) : IValueGetter {
public readonly int GetValue(int index) {
return index + 5;
}
}
internal class ClassValueGetter1 : IValueGetter {
public int GetValue(int index) {
return index + 5;
}
}
internal class ClassValueGetter2 : IValueGetter {
public int GetValue(int index) {
return index + 7;
}
}
internal static class Demo2 {
public static void Run() {
var t1 = new MyTestClass<StructValueGetter>(new StructValueGetter());
RunDemo("StructValueGetter ", t1.Run);
var t2 = new MyTestClass<ClassValueGetter1>(new ClassValueGetter1());
RunDemo("ClassValueGetter1 ", t2.Run);
var t3 = new MyTestClass<ClassValueGetter2>(new ClassValueGetter2());
RunDemo("ClassValueGetter2 ", t3.Run);
var t4 = new MyTestClass<IValueGetter>(new ClassValueGetter1());
RunDemo("IValueGetter-1 ", t4.Run);
var t5 = new MyTestClass<ClassValueGetter1>(new ClassValueGetter1());
RunDemo("ClassValueGetter1 ", t5.Run);
var t6 = new MyTestClass<StructValueGetter2>(new StructValueGetter2());
RunDemo("StructValueGetter2", t6.Run);
var t7 = new MyTestClass<IValueGetter>(new ClassValueGetter2());
RunDemo("IValueGetter-2 ", t7.Run);
var t8 = new MyTestClass<IValueGetter>(new StructValueGetter());
RunDemo("IValueGetter-3 ", t8.Run);
var t9 = Activator.CreateInstance(typeof(MyTestClass<>).MakeGenericType(typeof(StructValueGetter)), new StructValueGetter());
Action action9 = (Action)Delegate.CreateDelegate(typeof(Action), t9, t9.GetType().GetMethod("Run"));
RunDemo("Dynamic-Struct ", action9);
}
static void RunDemo(string caption, Action action) {
var stopWatch = Stopwatch.StartNew();
action();
stopWatch.Stop();
Console.WriteLine($"{caption} time = {stopWatch.Elapsed}");
}
}
}
Demo2.Run();
在.net 8.0 Release 编译执行的参考结果如下:
StructValueGetter time = 00:00:00.6920186
ClassValueGetter1 time = 00:00:01.1887137
ClassValueGetter2 time = 00:00:05.2889692
IValueGetter-1 time = 00:00:01.1652195
ClassValueGetter1 time = 00:00:01.1625259
StructValueGetter2 time = 00:00:00.6488674
IValueGetter-2 time = 00:00:05.2114724
IValueGetter-3 time = 00:00:07.1394676
Dynamic-Struct time = 00:00:00.6491220
结论
泛型编译特性对性能有所影响,我们发现:
- 泛型参数是 Struct 比 class 的性能要好,大约有两倍的差异;
- 泛型参数如果存在多个 Struct 可能时,性能没有影响,但如果泛型参数存在多个 class 可能时,性能急剧下降5倍之多;
- 泛型参数如果是接口形式,无论实际填充的结构还是类,其最终的执行性能一定是很慢的;
- 使用反射(例如:MakeGenericType)构建出的泛型实例,其实际运行性能并不受影响,非常适合高度定制的运行时类型构建,这一点非常重要,例如你可以在运行时检测实际情况,构建出不同的比较器对象,虽然构建的工厂方法返回的是接口,但你可以使用反射的方式动态传入字典的比较器参数(实际上c#的
Dictionary<TKey, TValue>这点设计是失败的,他的comparer不是一个泛型参数,而是接口);
综上所述,了解C#泛型编译特性对性能的影响是编写高性能代码的重要一部分,合理使用对于关键性代码性能至关重要。
C# 泛型编译特性对性能的影响的更多相关文章
- AOT和单文件发布对程序性能的影响
前言 这里先和大家介绍一下.NET一些发布的历史,以前的.NET框架原生并不支持最终编译结果的单文件发布(需要依赖第三方工具),我这里新建了一个简单的ASP.NET Core项目,发布以后的目录就会像 ...
- 高性能JavaScript-JS脚本加载与执行对性能的影响
在web产品优化准则中,很重要的一条是针对js脚本的加载和执行方式的优化.本篇文章简单描述一下其中的优化准则. 1. 脚本加载优化 1.1 脚本位置对性能的影响 优化页面加载性能的原则之一是将scri ...
- C++ 性能剖析 (四):Inheritance 对性能的影响
(这个editor今天有毛病,把我的format全搞乱了,抱歉!) Inheritance 是OOP 的一个重要特征.虽然业界有许多同行不喜欢inheritance,但是正确地使用inheritanc ...
- Git 2.7: 一个新的带来许多新特性和性能提升的主要版本
在2.6版本发布两个月之后,Git 2.7发布.该版本带来了许多新特性以及性能的提升. 本文选取了Git 2.7带来的主要变化: git remote支持get-url子命令,可以显示指定远端的URL ...
- JS脚本加载与执行对性能的影响
高性能JavaScript-JS脚本加载与执行对性能的影响 在web产品优化准则中,很重要的一条是针对js脚本的加载和执行方式的优化.本篇文章简单描述一下其中的优化准则. 1. 脚本加载优化 1.1 ...
- Angular2 VS Angular4 深度对比:特性、性能
欢迎大家持续关注葡萄城控件技术团队博客,更多更好的原创文章尽在这里~~ 在Web应用开发领域,Angular被认为是最好的开源JavaScript框架之一. Google的Angular团队已于3月 ...
- 一:MySQL数据库的性能的影响分析及其优化
MySQL数据库的性能的影响分析及其优化 MySQL数据库的性能的影响 一. 服务器的硬件的限制 二. 服务器所使用的操作系统 三. 服务器的所配置的参数设置不同 四. 数据库存储引擎的选择 五. 数 ...
- Redis基础用法、高级特性与性能调优以及缓存穿透等分析
一.Redis介绍 Redis是一个开源的,基于内存的结构化数据存储媒介,可以作为数据库.缓存服务或消息服务使用.Redis支持多种数据结构,包括字符串.哈希表.链表.集合.有序集合.位图.Hype ...
- Redis 宝典 | 基础、高级特性与性能调优
转载:Redis 宝典 | 基础.高级特性与性能调优 本文由 DevOpsDays 本文由简书作者kelgon供稿,高效运维社区致力于陪伴您的职业生涯,与您一起愉快的成长. 作者:kelgon ...
- 2019-8-31-C#-程序集数量对软件启动性能的影响
title author date CreateTime categories C# 程序集数量对软件启动性能的影响 lindexi 2019-08-31 16:55:58 +0800 2018-10 ...
随机推荐
- Xshell使用技巧及常用配置
Xshell使用 1.调整 Xshell 的终端显示和回滚缓冲区大小 磨刀不误砍柴工,为了更方便地学习 Linux,首先得对终端进行一些调整,步骤如下: 首先通过 xshell 顶部菜单中的文件--& ...
- 【python笔记】使用zip函数迭代多个可迭代对象
入门使用 # 示例代码 warframe = ["saryn", "wisp", "volt"] counts = [len(n) for ...
- Protobuf vs JSON
Protobuf(Protocol Buffers)和 JSON 都是数据序列化格式,但它们在许多方面有着显著的不同.以下是对两者的一些主要比较: 数据大小和速度: Protobuf:由于 Proto ...
- Java Maven POM配置参考
介绍 什么是POM? POM代表"项目对象模型".它是一个名为pom.XML的文件中保存的Maven项目的XML表示. 快速概览 这是一个直接位于POM项目元素下的元素列表.请注意 ...
- LeetCode952三部曲之三:再次优化(122ms -> 96ms,超51% -> 超91%)
欢迎访问我的GitHub 这里分类和汇总了欣宸的全部原创(含配套源码):https://github.com/zq2599/blog_demos 本篇概览 本文是<LeetCode952三部曲之 ...
- 《Python魔法大冒险》008 石像怪的挑战:运算符之旅
小鱼和魔法师继续深入魔法森林.不久,他们来到了一个巨大的魔法石圈旁边.石圈中心有一个闪闪发光的魔法水晶,周围则是一些神秘的符号.但令人意外的是,水晶的旁边还有一个巨大的石像怪,它的眼睛散发着红色的光芒 ...
- 谷粒商城微服务分布式高级篇:linux下使用docker安装ElasticSearch
[root@localhost ~]# docker pull elasticsearch:7.8.0 安装elasticsearch:7.8.0[root@localhost ~]# docker ...
- 使用 Sealos 构建低成本、高效能的私有云
这个时候谈论私有云似乎有点反直觉?大部分人认知不是上云是大趋势嘛?我也比较认可上云,不过私有云也是云,今天给大家带来一个新的选择 -- 用云,只需一个 Sealos 就够了. 看看我们怎么做到更低的成 ...
- html表单与框架
1.以form开头 其中常用的属性有 action="" method="" enctype="" name="" ...
- 2020/4/26 2-sat 学习笔记
2-sat 吧.... 其实我jio得它一点都不难 嗯 2-sat是个啥东西呢?其实就是有很多人,他们每个人有两个要求,一个要求可以说是要求一个数为0或1而对于第i个数,我们可以选择为0或为1最终询问 ...