title author date CreateTime categories
C# 7.2 通过 in 和 readonly struct 减少方法值复制提高性能
lindexi
2018-12-25 9:24:6 +0800
2018-11-24 16:40:58 +0800
C#

在 C# 7.2 提供了一系列的方法用于方法参数传输的时候减少对结构体的复制从而可以高效使用内存同时提高性能

在开始阅读之前,希望读者对 C# 的值类型、引用类型有比较深刻的认知。

在 C# 中,如果对内存有严格的要求,同时需要减少 GC 的情况,推荐此时使用结构体。但是结构体有一个缺点在于,结构体在每次调用方法作为参数传递的时候都会新建一个副本,这对于性能要求特别高的情况是不适合的。

定义一个值类型

  1. struct Int256
  2. {
  3. public Int256(long bits0, long bits1, long bits2, long bits3)
  4. {
  5. Bits0 = bits0;
  6. Bits1 = bits1;
  7. Bits2 = bits2;
  8. Bits3 = bits3;
  9. }
  10.  
  11. public long Bits0 { set; get; }
  12. public long Bits1 { get; }
  13. public long Bits2 { get; }
  14. public long Bits3 { get; }
  15. }

此时通过一个简单的赋值就可以获取复制

  1. Int256 f1 = new Int256(0, 1, 2, 1);
  2. var f2 = f1;
  3. f2.Bits0 = 2;
  4. Console.WriteLine($"f1.bits0={f1.Bits0} f2.bits0={f2.Bits0}");
  5. //f1.bits0=0 f2.bits0=2

在调用方法的时候也一样,传入参数就是复制一个新的值

  1. static void Main(string[] args)
  2. {
  3. Int256 f1 = new Int256(0, 1, 2, 1);
  4. Foo(f1);
  5. Console.WriteLine($"{f1.Bits0}"); // 0
  6. }
  7.  
  8. private static void Foo(Int256 foo)
  9. {
  10. foo.Bits0 = 2;
  11. }

对于很小的值类型,如果小于 IntPtr.Size 的传输,会比引用传递的复制速度快,但是对比比较大的值类型,如上面定义的,复制一次需要的时间会比较长

特别是存在很多次的值传递的时候,如下面的代码,会调用 1000 次的值传递。除了性能的问题,还存在堆栈的内存的问题

定义一个很大的值类型,里面包含 10000 个 double 看起来就很大

  1. struct Double10000
  2. {
  3. public double Double0 { get; }
  4. public double Double1 { get; }
  5. public double Double2 { get; }
  6. ……
  7. public double Double9999 { get; }
  8. }

用递归的方式进行调用,运行的时候很快就可以看到堆栈都被申请的值传递使用,同时 CPU 的使用很高

  1. static void Main(string[] args)
  2. {
  3. Double10000 foo = new Double10000();
  4. Foo(foo);
  5. }
  6.  
  7. private static void Foo(Double10000 foo, int n = 100)
  8. {
  9. if (n == foo.Double0)
  10. {
  11. return;
  12. }
  13.  
  14. Foo(foo, n - 1);
  15. }

如果可以让值类型和引用一样传递,是不是就可以减少值类型的复制同时减少堆栈的使用,请注意不要纠结值类型是分配在堆中还是栈中的问题,上面的代码更多的是方法的递归

对比内存的使用,更多的时候关心的是运行的速度。添加一些代码用来测试性能,同时减少调用

  1. var st = new Stopwatch();
  2. st.Start();
  3. Foo(foo);
  4. st.Stop();
  5. Console.WriteLine(st.ElapsedTicks);
  1. private static void Foo(Double10000 foo, int n = 10)
  2. {
  3. if (n == foo.Double0)
  4. {
  5. return;
  6. }
  7.  
  8. Foo(foo, n - 1);
  9. }

这里输出的 ElapsedTicks 的单位是 100ns 需要知道 1ms=1000000ns 也就是 1w 的 tick 就是 1 毫秒,下面我运行 3 次代码,收集到的值

  1. 10991
  2. 14950
  3. 16183

在 C# 7.2 可以使用 in 关键字告诉 VisualStudio 当前的方法不会对传进来的结构体进行修改,当前这样写只是语法层面。如果有一些厉害的黑客,可能还继续这样写入,于是为了防止真的进行修改,在底层还是复制了一份。

也就是只是在参数里面使用了 in 是不够的,具体请看这个拖后腿的“in” - Bean.Hsiang - 博客园

如果想要更好的使用内存同时提高性能,只有在可以被标记为只读的结构体的时候使用 in 才可以

先将 Double100 标记为 readonly 如果一个值类型标记为 readonly 也就无法对里面的字段或属性进行设置了

在 Foo 传入的方法参数标记 in 这样就完成了,因为 in 表示对参数不进行修改,而传入的是 readonly struct 本来就不能被修改,于是就传入 struct 的引用

  1. readonly struct Double10000
  2.  
  3. private static void Foo(in Double10000 foo, int n = 10)
  4. {
  5. if (n == foo.Double0)
  6. {
  7. return;
  8. }
  9.  
  10. Foo(foo, n - 1);
  11. }

同样运行 3 次,可以看到速度是原来的 10 倍

  1. 2052
  2. 1837
  3. 1683

同时占用的堆栈更小,可以使用更多的递归,修改 Foo 函数调用次数为 1000 可以看到还能运行,但是如果去掉了参数 in 最多只能调用 20 次

没有加 in 的参数,运行了 17 次

添加了 in 之后因为不需要复制值,减少内存的时候,此时运行了 1000 次递归都可以,在使用in之后速度和使用内存都比较好

在很多次方法调用使用参数的时候,如果传入的值是值类型,如果此时的 struct 里面的属性都是只读属性,推荐将 struct 标记为 readonly 同时在方法参数标记 in 这样可以让 struct 作为引用传递,也就是复制的只是指针,只要 struct 的长度比指针小就推荐这个方法

2018-12-25-C#-7.2-通过-in-和-readonly-struct-减少方法值复制提高性能的更多相关文章

  1. 2018.12.25 SOW

    1. Understanding Customer Requirements 11.1. Project Overview 21.2. System Requirements 21.3. Indust ...

  2. 2018.12.25 Spring中JDBCTemplate模版API学习

    1 Spring整合JDBC模版 1.1 spring中土拱了一个可以操作数据库的对象.对象封装了jdbc技术 JDBCTemplateJDBC模板对象 1.2 与DBUtils中的QueryRunn ...

  3. 12.25模拟赛T1

    可以区间dp,但是复杂度太高. 所以应该是贪心,怎么贪心呢? 这种题目,最好还是手玩找一些规律. 可以发现,由于保证可以m次填完,所以颜色之间没有相互包含关系. 比较像分治的模型. 所以考虑拿到一个区 ...

  4. loli的测试-2018.12.9

    模拟赛-2018.12.9 这是NOIP之后第一次模拟赛...但是考的比较悲惨. 非常喜欢写考试总结,不知道为什么... T1:https://www.luogu.org/problemnew/sho ...

  5. 2018.12.02 Socket编程之初识Socket

    Socket编程主要分为TCP/UDP/SCTP三种,每一种都有各自的优点,所以会根据实际情况决定选用何种Socket,今天开始我将会逐步学习Socket编程,并将学习过程记录于此. 今天学习的是TC ...

  6. OPPO Developers Conference(2018.12.26)

    时间:2018.12.26地点:北京国家会议中心

  7. Tencent Cloud Developers Conference(2018.12.15)

    时间:2018.12.15地点:北京朝阳悠唐皇冠假日酒店

  8. 2018.12.1 Test

    目录 2018.12.1 Test A 串string(思路) B 变量variable(最小割ISAP) C 取石子stone(思路 博弈) 考试代码 B C 2018.12.1 Test 题目为2 ...

  9. 2018/04/25 基于 编译安装的 PHP7 安装 swoole 扩展

    在上一篇文章我们知道了如何去编译安装一个自己需要的 PHP 版本. 2018/04/25 PHP7的编译安装 这里还没有完,我们还需要安装我们的扩展,才算完成今天的任务. -- 下载扩展 还是官网下载 ...

  10. 「版本升级」MyEclipse CI 2018.12.0正式发布

    新版本MyEclipse为WildFly 14新增一个新的服务器连接器,改进性能并新增一些Java 10修复程序.新版本为IDE做了几个核心修复,这是MyEclipse 2018一个更棒的升级. [M ...

随机推荐

  1. git分布式版本控制系统权威指南学习笔记(四):git reset

    文章目录 git reset目录树重写 git reset 重置 git reset目录树重写 git reset --soft 暂存区工作区不变 git reset --hard git reset ...

  2. iptables默认规则

    iptables默认规则 *filter :INPUT ACCEPT [0:0] :FORWARD ACCEPT [0:0] :OUTPUT ACCEPT [34:4104] -A INPUT -m ...

  3. js登陆验证错误不刷新页面

    验证函数返回 false;返回到onclickonclick 其实也是一个函数.. 所以需要加 return;

  4. java-day23

    事务的四大特征: 1.原子性:是不可分割的最小操作单位,要么同时成功,要么同时失败. 2.持久性:当事务提交或回滚后,数据库会持久化的保存数据. 3.隔离性:多个事务之间,相互独立. 4.一致性:事务 ...

  5. hdu6311 /// 欧拉路径 无向图最小路径覆盖 输出正反路径

    题目大意: 给定n m 为图的点数和边数 接下来m行 u v 为u到v有一条边 要求最少几笔能画完图的所有边 输出每笔画过的路径编号 正数编号正向 负数编号反向 题解:https://www.cnbl ...

  6. jsk

    题目描述 码队的女朋友非常喜欢玩某款手游,她想让码队带他上分.但是码队可能不会带青铜段位的女朋友上分,因为码队的段位太高(已经到达王者),恐怕不能和他的女朋友匹配游戏. 码队的女朋友有些失落,她希望能 ...

  7. java oop第15章_Socket网络编程

    一.   TCP/IP协议(Transmission Control Protocol/Internet Protocol)传输控制协议/Internet协议,是通信领域的基础.核心协议, 其他的协议 ...

  8. 上线出现[x86_64, i386]

    echo "Target architectures: $ARCHS" APP_PATH="${TARGET_BUILD_DIR}/${WRAPPER_NAME}&quo ...

  9. TRUNCATE - 清空一个表

    SYNOPSIS TRUNCATE [ TABLE ] name DESCRIPTION 描述 TRUNCATE 快速地从一个表中删除所有行.它和无条件的 DELETE 有同样的效果,不过因为它不做表 ...

  10. docker Dockerfile学习---构建mongodb环境

    1.创建项目目录并上传包 mkdir centos_mongodb cd centos_mongodb .tgz 2.编辑配置文件 vi mongodb.conf dbpath = /data/usr ...