C#中ref和out的原理
去年在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的原理的更多相关文章
- C#中ref和out的使用与区别
C#中ref关键字和out关键字所实现的功能差不多,都是指定一个形参按照引用传递而不是实参的副本传递.但是二者适用场景还是有些区别的:out适合用在需要retrun多个返回值的地方,而ref则适合用在 ...
- 学习重点:1、金典的设计模式在实际中应用2、JVM原理3、jui源代码
学习重点:1.金典的设计模式在实际中应用 2.JVM原理 3.jui源代码
- C#中ref和out的区别浅析
这篇文章主要介绍了C#中ref和out的区别浅析,当一个方法需要返回多个值的时候,就需要用到ref和out,那么这两个方法区别在哪儿呢,需要的朋友可以参考下 在C#中通过使用方法来获取返回值时,通 ...
- Vue.js-11:第十一章 - Vue 中 ref 的使用
一.前言 在之前的前端开发中,为了实现我们的需求,通常采用的方案是通过 JS/Jquery 直接操纵页面的 DOM 元素,得益于 Jquery 对于 DOM 元素优异的操作能力,我们可以很轻易的对获取 ...
- Spring中EmptyResultDataAccessException异常产生的原理及处理方法
Spring中EmptyResultDataAccessException异常产生的原理及处理方法 Spring中使用JdbcTemplate的queryForObject方法,当查不到数据时会抛出如 ...
- Fastjson-fastjson中$ref对象重复引用问题:二
import java.util.ArrayList; import java.util.List; import com.alibaba.fastjson.JSON; import com.alib ...
- Fastjson-fastjson中$ref对象重复引用问题
当你有城市数据,你需要按国内.国际.热门城市分成数组的形式给出并输出为json格式. 第一个问题,你的数据格式,需要按字母类别划分,比如: "int": { "C&quo ...
- React中ref的使用方法
React中ref的使用方法 在react典型的数据流中,props传递是父子组件交互的唯一方式:通过传递一个新的props值来使子组件重新re-render,从而达到父子组件通信.当然,就像reac ...
- 基于接口回调详解JUC中Callable和FutureTask实现原理
Callable接口和FutureTask实现类,是JUC(Java Util Concurrent)包中很重要的两个技术实现,它们使获取多线程运行结果成为可能.它们底层的实现,就是基于接口回调技术. ...
随机推荐
- zookeeper实现的分布式锁
在分布式系统中,多个jvm对共享资源进行操作时候,要加上锁,这就是分布式锁 利用zookeeper的临时节点的特性,可以实现分布式锁 public class ZookeeperDistrbuteLo ...
- 关于于c++中的类型转换
隐藏式类型转换 void test() { ; ; a = b; //此时发生的是默认的类型转 //(据说编译器是微软的编译器是不允许编译通过) std::cout << a <&l ...
- LeetCode第154场周赛(Java)
估计要刷很久才能突破三道题了.还是刷的太少.尽管对了前两题,但是我觉得写的不怎么样.还是将所有题目都写一下吧. 5189. "气球" 的最大数量 题目比较简单.就是找出一个字符串中 ...
- Source Insight4.0软件破解版
安装source insightt4.0 1.将下载好的sourceinsight4.exe替换安装在program file(x86)目录下的sourceinsight4.exe; 2.启动sour ...
- Tcl语言学习--基础知识
一.脚本.命令和单词符号 一个TCL脚本可以包含一个或多个命令.命令之间必须用换行符或分号隔开. 1.关键字/变量 变量是程序的基础变量组成:变量名.变量值变量名要求:任何字符串都可以作为变量名,区分 ...
- laravel 自定义验证 Validator::extend
laravel 自定义验证 $messages = [ 'name.integer' => '名字不能为整型', 'name.max' => '长度不能超过5', ]; public st ...
- JMeter安装及简单应用示例
一.Jmeter下载 官网地址:http://jmeter.apache.org/ 1.进入官网 2.选中一个版本下载 3.解压安装即可 二.Jmeter环境变量配置 1. 电脑桌面----> ...
- C#获取文件夹下的所有文件的方法
目录 #基础知识 #只获取目录下一级的文件夹与文件 # 递归地输出当前运行程序所在的磁盘下的所有文件名和子目录名 正文 #基础知识 1.获得当前运行程序的路径 1 string rootPath ...
- Java自学-控制流程 结束外部循环
Java中结束外部循环 Java中如何结束外部for循环? 示例 1 : 结束当前循环 break; 只能结束当前循环 public class HelloWorld { public static ...
- 【开发笔记】- 转义html特殊字符
package com.juihai.util; import org.apache.commons.lang.StringUtils; import org.springframework.web. ...