去年在CSDN上写的,现在把它搬过来。

一、引发问题

用了那么久的 ref out ,你真的了解它们是如何使得实参与形参的值保持同步的吗?

二、研究前提

要研究这个问题,前提是要了解 C# 中方法间参数是如何传递的:

1.CLR支持两种类型:值类型引用类型

a. 值类型:值一般保存在线程栈上,作为类对象的字段时保存在堆上。

b. 引用类型:对象实例保存在堆上,引用保存在线程栈上,值类型可以通过装箱变为引用类型。

  

 //表示引用类型
class Ref
{
private int _x;
public int X
{
get => _x;
set
{
_x = value;
}
}
} static void TestValAndRef()
{
//第一部分
int a1 = ;
var ref1 = new Ref()
{
X =
}; //第二部分
int a2 = a1;
a2 = ;
var ref2 = ref1;
ref2.X = ;
}

上述代码执行时变量的存储情况:

2.参数传递方式分为传值传引用两种。

3.对于CLR来说,使用out和ref都会生成相同的IL代码,并且元数据除了一个bit(用于记录声明方法时指定的是out还是ref)外,完全一致。

 

 //测试ref
static void TestRef(ref Ref r)
{
r = new Ref()
{
X = -
};
} //测试out
static void TestOut(out Ref r)
{
r = new Ref()
{
X = -
};
} static void Main(string[] args)
{
var ref1 = new Ref()
{
X =
}; TestRef(ref ref1);
TestOut(out Ref ref2);
}

以上代码编译出来的IL为:

可以看到,TestRef和TestOut方法对应的IL完全相同!

4.在CLR中,方法的参数以及返回值都是通过栈来保存的,这些形参虽然表示的东西和实参看起来时一致的,但是实际上是分开存储的,即形参和实参是两个不同的变量。

三、研究问题

1.CLR默认所有方法参数传递方式都是传值:

a.对于值类型来说,传递的是值的副本。例如线程栈中 a1 的值:5。

b.对于引用类型来说,传递的是对象的引用,而引用本身是传值的,调用方法内用形参把引用存起来,如果在调用方法内部更改了形参内保存的引用(new一个新对象或用对其赋另一个对象),那么该形参就与实参断了联系,随后的修改对实参不起作用;但如果引用未被改变的情况下进行了更改,实际上就是对实参进行的更改。例如线程栈中 ref1 的值:类型对象的引用。

2.当使用了ref或out后,C#传值方式就变为了传引用,类似于 C 中的 &a1,我想这里的&就是对应的ref和out吧:

a.对于值类型来说,传递的是对值的引用(可以理解为值的地址,类似于引用类型的传值方式)=> &形参,去掉&,剩下的形参实际上就是实参,所以这个形参中保存的引用永远不会被改变,也就是始终更改的是实参的值。例如对线程栈中 a1 的引用。

b.对于引用类型来说,传递的是对变量的引用(可以理解为指向实例对象引用的栈地址的引用,通俗的讲就是对象的引用是保存在栈的某个地址上,这里传递的就是对于该地址的引用)=>  &形参,这样就保证了调用方法内部使用的就是实参对象,而不是其引用的副本,所以任何更改都是对实参进行更改的。例如对线程栈中 ref1 的引用。

疏漏之处在所难免,如果有理解不对的地方请在下方留言,谢谢!

C#中ref和out的原理的更多相关文章

  1. C#中ref和out的使用与区别

    C#中ref关键字和out关键字所实现的功能差不多,都是指定一个形参按照引用传递而不是实参的副本传递.但是二者适用场景还是有些区别的:out适合用在需要retrun多个返回值的地方,而ref则适合用在 ...

  2. 学习重点:1、金典的设计模式在实际中应用2、JVM原理3、jui源代码

    学习重点:1.金典的设计模式在实际中应用 2.JVM原理 3.jui源代码

  3. C#中ref和out的区别浅析

    这篇文章主要介绍了C#中ref和out的区别浅析,当一个方法需要返回多个值的时候,就需要用到ref和out,那么这两个方法区别在哪儿呢,需要的朋友可以参考下   在C#中通过使用方法来获取返回值时,通 ...

  4. Vue.js-11:第十一章 - Vue 中 ref 的使用

    一.前言 在之前的前端开发中,为了实现我们的需求,通常采用的方案是通过 JS/Jquery 直接操纵页面的 DOM 元素,得益于 Jquery 对于 DOM 元素优异的操作能力,我们可以很轻易的对获取 ...

  5. Spring中EmptyResultDataAccessException异常产生的原理及处理方法

    Spring中EmptyResultDataAccessException异常产生的原理及处理方法 Spring中使用JdbcTemplate的queryForObject方法,当查不到数据时会抛出如 ...

  6. Fastjson-fastjson中$ref对象重复引用问题:二

    import java.util.ArrayList; import java.util.List; import com.alibaba.fastjson.JSON; import com.alib ...

  7. Fastjson-fastjson中$ref对象重复引用问题

    当你有城市数据,你需要按国内.国际.热门城市分成数组的形式给出并输出为json格式. 第一个问题,你的数据格式,需要按字母类别划分,比如: "int": { "C&quo ...

  8. React中ref的使用方法

    React中ref的使用方法 在react典型的数据流中,props传递是父子组件交互的唯一方式:通过传递一个新的props值来使子组件重新re-render,从而达到父子组件通信.当然,就像reac ...

  9. 基于接口回调详解JUC中Callable和FutureTask实现原理

    Callable接口和FutureTask实现类,是JUC(Java Util Concurrent)包中很重要的两个技术实现,它们使获取多线程运行结果成为可能.它们底层的实现,就是基于接口回调技术. ...

随机推荐

  1. Mysql 索引基础

    [1]什么是索引?为什么要建立索引? 索引,其实就是目录. 索引,用于快速找出在某个列中有某个特定值的行. 不使用索引,MySQL必须从第一条记录开始查找整张表,直到找出相关的行,那么表越大,查询数据 ...

  2. 在CAD中插入谷歌地球卫星地图

    本文主要介绍如何在CAD中插入谷歌地球卫星地图,作为参照光栅图像.谷歌地球卫星地图使用“迈高图-地图数据下载器”(以下简称:迈高图)下载.迈高图会给出相关插入参数(插入点和缩放比例),保证插入卫星地图 ...

  3. C++四大特性之封装

    C++四大特性 C++作为面向对象编程语言,具备面向对象编程(Object Oriented Programming,OOP,面向对象程序设计)的四大特性.抽象,封装,继承,多态. 所谓抽象,就是对具 ...

  4. python教程:用简单的Python编写Web应用程序

    python现在已经成为很多程序员关注的编程语言之一,很多程序员也都开始弄python编程,并且很多时候都会用自己的操作来选择,而现在不管是程序员还是少儿编程,都会有python这门课,今天就和大家分 ...

  5. emmet 配置文件

    snippets.json(添加自己的或更新现有的片段) preferences.json(更改某些Emmet过滤器和操作的行为) SyntaxProfiles.json(定义生成的HTML / XM ...

  6. 「UER#2」谣言的传播

    「UER#2」谣言的传播 写了个乱搞,怎么莫名其妙就AC了,这...,之后又想了30min结合题解终于会证了. 首先最大值比较简单,记 \(f_i\) 为第 \(i\) 个点能到达的点数,上界 \(\ ...

  7. 表单提交学习笔记(一)—利用jquery.form提交表单(后台.net MVC)

    起因:一开始想用MVC本身的Form提交方法,但是提交完之后想进行一些提示,MVC就稍显不足了,最后用jquery插件---jquery.form.js,完美解决了问题~~ 使用方法 一.下载jque ...

  8. echarts统计x轴区间的数值

    有时我们需要统计自定义echarts图,统计x轴区间的y轴数量. 思路是利用echarts的自定义配置:option.series[i].type='custom'中的renderItem(param ...

  9. Java 之 Cookie

    Cookie 一.会话概述 1.会话:一次会话中包含多次请求和响应.       一次会话:浏览器第一次给服务器资源发送请求,会话建立,直到有一方断开为止. 2.功能: 在一次会话的范围内的多次请求间 ...

  10. scrapy 爬虫中间件-offsite和refer中间件

    环境使用anaconda 创建的pyithon3.6环境 mac下 source activate python36 mac@macdeMacBook-Pro:~$ source activate p ...