C#7.0 ref引用传递

1.概要

在工作中大家用到引用类型是非常多的,大家都知道引用类型在使用过程中传递的是对象引用并不会发生整个对象复制。而值类型在传递的过程中就不一样了,我曾经在编写代码时希望通过值类型来压低应用程序的内存占用,在高并发的情况大量的对象需要在程序里流转这个时候看内存监测的时候会发现,内存并没有变少。虽说值类型能提供很多好处但有一个缺点就是会发生复制,那么如何规避复制这个缺点呢?我们往下看(本文只是简略的分享,实际上要把这一整块讲明白几千字是远远不够的)。

2.详细内容

2.1 ref

在继续了解C#7.0的ref特性需要了解一些前置知识点,首先是变量和值的区别。

(下图)变量包含内容:

  • 变量名称

  • 编译时类型

  • 当前值

2a7549fd64faec09a6c9ba2c1656a263.png

变量的声明本质是在内存中开辟一段内存空间,给变量x赋值相当于是覆盖了之前的值。当变量类型是引用类型时,控件里的值不再是对象本身。而是对象的引用,就是通过内存地址找到对象。(如果加上ref关键字,ref的引用和对象引用是不同的概念。通过值传递对象引用和通过引用传递变量是不同的。)

当把某个变量值复制给另外一个变量时,只是这个值本身发生了复制。这两个变量依然是独立的,之后任何一个变量的值修改不会影响另外一个变量。

4767aeac698284a11af8e70476b0a1a1.png

这种方式的值复制,和调用方法时对值参数的操作是相同的:方法实参的值被复制到了另一个新的空间中。

1a557a0d961d4c99b9cf2bdbf9f7a8c6.png

而ref参数的行为与此不同。使用ref参数,不会创建开辟新的空间,而是调用放提供一个现有的包含初始值的空间。可以理解为一个空间同时被两个地址指向:一个是调用方使用的该变量的表示,另一个是形参的名称。

d6d94a64d0d3026e872a3caf33c0f18b.png

如果在方法中修改了ref参数的值,即修改了纸上的现有值。当方法返回时,修改的结果就会反回给调用方,因为修改的是同一个命名空间的值。

2.2 ref readonly

前面提到的变量都是可写变量,以下两个独立场景中,只允许ref可写变量就显得有些不足了。

  • 可能需要给某个只读字段添加引用地址,避免复制以提升效率。

  • 可能需要只允许通过ref变量进行只读访问。

C#7.2加入了ref readonly解决了上述问题。ref局部变量可以使用readonly进行修饰,得到的结果自然是只读的,就像只读字段一样。不能为只读变量赋新值,如果它是结构体,则不能修改任何字段或者调用属性的setter方法。

代码示例:

public readonly struct Juster{public int Age { get; }public Juster(int age) => Age = age;}

使用readonly的两处都需要协作:如果调用一个带有ref readonly返回的方法或者索引器,并且需要将结果保存到一个局部变量中,那么这个局部变量必须由ref readonly修饰。

internal class Class1{static readonly int field = DateTime.UtcNow.Second;static ref readonly int GetJusterTime() => ref field;public void Test() {ref readonly int local = ref GetJusterTime();Console.WriteLine(local);}}

2.3 in参数

C#7.2为方法参数加入了in修饰符,该修饰符的使用方式与ref、out相同,但目的不同。一个带有in修饰符参数,可以通过引用传递避免复制提升效率,同时可以保证参数值不被修改。在方法内部,in参数的行为类似于ref readonly局部变量。该变量依然是由调用方传入的一个内存地址,因此要保证方法不会被修改值,否则修改结果回影响调用方,这样就违背了in参数的意义。

安全的使用使用in参数:

public static double PublicMethod(Juster juster) {double result = GetScale(in juster);return result + result;}private static double GetScale(in Juster input) => input.Age * input.Age;

这种方式可以防止参数被意外修改,因为方法是私有的,我们可以检查所有调用方,确定它们不会传递哪些在方法执行时可能被修改的参数。在方法GetScale被调用时,每个结构体只会被复制一次,复制之后私有方法调用时都是别名。这样就把自己的代码和其他线程中调用方的任何修改,或者其他方法的副作用隔离开来了。

使用建议:

  • 只有确定性能提升客观,采用in参数,例如使用大型结构体。

  • 在公共api中尽量避免使用in参数,除非即便参数值发生变化,方法也能正确执行。

  • 可以考虑通过公共方法作为防止参数被修改的外部屏障,然后再内部私有方法中使用in参数来减少复制。

  • 对于采用in参数的方法,在调用时考虑显式给出in修饰符。

性能相关测试

https://cloud.tencent.com/developer/article/1402446

https://www.cnblogs.com/BeanHsiang/p/8687780.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/281322.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

android面试详解

前台就是和用户交互的进程 可见进程例如一个activity被一个透明的对话框覆盖,该activity就是可见进程 服务:service进程 后台一个activity按了home按键就是从前台退回到后台 标准模式:不管任务栈是否存在相同的activity都会创建一个新的activ…

element-ui Notification重叠问题,原因及解决办法

在1个方法中调用两次this.$notify方法,会出现通知框重叠的问题 methods: {checkLogin: function () {if (this.username ) {this.$notify({title: 提示,message: 请输入用户名})}if (this.password ) {this.$notify({title: 提示,message: 请输入用户密码})}}}网上…

Visual Stiudio使用技巧

技巧1 自动生成带参构造函数当我们在编写代码时会经常遇到初始化一个的类,需要通过构造函数进行对象初始化。那么这个时候我们可能会需要逐个去手动写,这样的工作即重复又无趣。如果是在项目非常紧急的情况下还有大量的字段需要与入参一一对应起来简直太…

C#性能测试BenchmarkDotnet

1.简介在我们开发高性能代码时,需要各种针对性能优化进行编码。那么如何才能知道我们所加的代码是否有性能方面的正向优化呢?有了BenchmarkDotNet,做性能对比测试就非常容易了,只需要把你的测试方法加上特性[Benchmark], 想做不同…

Requests获取连接的IP地址

在接口自动化的时候,需要获取到连接的本地IP地址,方法如下 1 import requests 2 3 rsp requests.get("http://www.baidu.com", streamTrue) 4 print rsp.raw._connection.sock.getpeername()[0] 5 print rsp.raw._connection.sock.getsockna…

阿里云APP(V4.3) SSH远程登录功能设置操作指南

阿里云APP V4.3 发布了,这次的升级,不仅在iOS和android平台上支持SSH远程登录ECS功能,也支持密钥登录哦~~~ SSH远程登录,这是一个连阿里巴巴自己的技术人员都开心不已的功能! 各位攻城狮们,从更新到V4.3的那…

JS专题之节流函数

本文共 2000 字,读完只需 8 分钟上一篇文章讲了去抖函数,然后这一篇讲同样为了优化性能,降低事件处理频率的节流函数。 一、什么是节流? 节流函数(throttle)就是让事件处理函数(handler&#xf…

【Flutter教程】从零构建电商应用(一)

在这个系列中,我们将学习如何使用google的移动开发框架flutter创建一个电商应用。本文是flutter框架系列教程的第一部分,将学习如何安装Flutter开发环境并创建第一个Flutter应用,并学习Flutter应用开发中的核心概念,例如widget、状…

为OWA自定义快捷键

这篇短文分享一下如何为自己常用的网页添加自定义功能,例如添加快捷键。我这里用一个常用的网站作为范例。它是Outlook Web Access (OWA), 它的地址一般如下。我在写邮件时希望能用一些快捷键来提高工作效率,但系统默认自带的快捷键特别少,而…

数据结构 快速排序

快速排序是对冒泡排序的一种改进,是所有内部排序算法中平均性能最优的排序算法。其基本思想是基于分治法的:在待排序数组L[1...n]中任取一个元素pivot作为基准,从数组的两端开始扫描。设两个指示标志(low指向起始位置,…

小米人员架构调整:组建中国区,王川任总裁

12月13日上午,小米内部发布人员调整公开信,信中传达了两个重要内容:将销售与服务部改组为中国区,任命集团高级副总裁王川兼任中国区总裁。 在今年9月份,也就是小米上市前夕,雷军在一封内部信中宣布对公司组…

Java基础 五 方法

方法 1.1 方法概述 在我们的日常生活中,方法可以理解为要做某件事情,而采取的解决办法。 如:小明同学在路边准备坐车来学校学习。这就面临着一件事情(坐车到学校这件事情)需要解决,解决办法呢&#xf…

附近有什么?8款可以查周边的App

如今科技发达的时代,手机的功能不仅仅只是能通讯聊天,而是逐渐的走进了人们的生活中。因为有了APP,我们的生活才更丰富,并且有很多是我们生活中不可缺少的软件,而这些软件便是根据手机中的GPS定位系统而来的。简单来说…

MyEclipse小问题与汉字处理

今天在使用MyEclipse时,遇到工作目录报错(如上图),解决方法如下:找到对应工作区(查看工作区的方法为:单击File → Switch Workspace 即可)依次打开 .metadata文件夹 → .plugins文件夹 → org.eclipse.core.runtime文件夹 → .set…

java B2B2C springmvc mybatis电子商务平台源码-消息队列之RocketMQ

RocketMQ出自阿里公司的开源产品,用 Java 语言实现,在设计时参考了 Kafka,并做出了自己的一些改进,消息可靠性上比 Kafka 更好。RocketMQ在阿里集团被广泛应用在订单,交易,充值,流计算&#xff…

VSCode同步设置

2022/4/1 更新 刚刚发现还有人在看这篇文章,这里更新一下,VSCode 从1.48版本开始已经内置了同步功能,可以不用再使用Settings Sync插件了。 点击左下角的用户或者设置的 Sign in to Sync Setting,使用GitHub或者Microsoft账户登…

配置三台服务器组成的ELK集群(二)

上一篇里主要是介绍了ES和ES-Head的安装过程,这一篇继续介绍ELK集群的其他核心组件安装过程。 五、安装Logstash: 本案的Logstash安装在10.113.130.117上;燃鹅,Logstash也可以利用多台组成集群,如果未来单台处理不过来…

Discuz X3.2源码解析 discuz_application类(转自百度)

discuz_application在/source/class/discuz/discuz_application.php中。 discuz_application继承自抽象类discuz_base discuz_application主要实现对运行环境、配置、输入、输出、数据库、设置、用户、session、移动模块、计划任务、手机预览等方面的初始化。 instance()函数来…

.NET性能优化-是时候换个序列化协议了

计算机单机性能一直受到摩尔定律的约束,随着移动互联网的兴趣,单机性能不足的瓶颈越来越明显,制约着整个行业的发展。不过我们虽然不能无止境的纵向扩容系统,但是我们可以分布式、横向的扩容系统,这听起来非常的美好&a…

Kubernetes-基于Helm安装部署高可用的Redis

1、Redis简介 Redis是一个开放源代码(BSD许可证)的代理,其在内存中存储数据,可以代理数据库、缓存和消息。它支持字符串、散列、列表、集合和位图等数据结构。Redis 是一个高性能的key-value数据库, 它在很大程度改进了…