摘一段来自官网的说明 :方法的参数列表中使用 ref 关键字时,它指示参数按引用传递,而非按值传递。 ref 关键字让形参成为实参的别名,这必须是变量。 换而言之,对形参执行的任何操作都是对实参执行的。

大体意思就是将实惨的引用作为参数传递,如果入参不加ref修饰,本身传递的是实参的值到方法中。

那什么是值,什么是引用?大体可以理解为堆栈的区别,在.net中大多数实例存在于托管堆栈中。struct,int32,int64,double,enum等数据类型声明的实例存在栈中,而string,class,delegate等存在于堆中。前者一般称为值类型,后者则叫做引用类型,那么引用类型和值类型在内存中的大概图如下:



可以看到引用类型被存储的时候真正内容存在堆中,而在stack中存储了一个堆中地址的引用指向堆中真正内容。

ref修饰入参的常用场景

当我们希望一个已有返回值的方法能够修改我一个外部基本类型的值的时候,我们可以将该参数加上ref关键字作为入参。具体原理其实就是将栈中具体的值替换为了栈的引用,说白了就是地址,幻想一下本来你高考作弊只是后台改了一下自己成绩和学霸一样,学霸的人生没有受到影响,现在你干脆把学霸的试卷改成了自己名字,学霸直接被你影响只能进厂打螺丝了,当然这个比喻不是很贴切,大家理解到意思就行。

那么为什么引用类型的入参我们不需要要添加ref?因为从上图我们能看出来栈中存储的本来就是引用类型的地址,所以引用类型不需要添加ref关键字,当你在方法内部修改了入参的一些属性值,外面的实惨依旧会受到影响。

引用类型添加ref的作用是啥?

我们在实际开发中还是能够碰到一些引用类型添加ref的场景。其实道理也是一样的,就是将引用类型的栈的地址传递到了方法中,那么和不添加有啥区别?我们来看看下面的代码:

Student student1 = new Student("Jack");
Student student2 = new Student("Lucy");
WithoutRef(student1);
WithRef(ref student2);
Console.WriteLine($"不带ref的方法---{student1.Name}");
Console.WriteLine($"带ref的方法---{student2.Name}"); Console.Read();
//不带ref的方法
static void WithoutRef(Student stu)
{
stu.Name = "Bruce";
} //不带ref的方法
static void WithRef(ref Student stu)
{
stu.Name = "Bruce";
} public class Student
{
public Student(string name)
{
Name = name; }
public string Name { get; set; }
}



可以看到结果一样的,两个实例的名字都变了

那我们再看看下面的代码:

Student student1 = new Student("Jack");
Student student2 = new Student("Lucy");
WithoutRef(student1);
WithRef(ref student2);
Console.WriteLine($"不带ref的方法---{student1.Name}");
Console.WriteLine($"带ref的方法---{student2.Name}"); Console.Read();
//不带ref的方法
static void WithoutRef(Student stu)
{
stu = new Student("Bruce");
} //不带ref的方法
static void WithRef(ref Student stu)
{
stu = new Student("Bruce");
} public class Student
{
public Student(string name)
{
Name = name; }
public string Name { get; set; }
}



我们发现第一个不带ref的方法名字没有发生变化。

代码的变化就是第二个例子重新new了一下,我们根据上面的原理分析就能知道为啥。由于ref将引用类型的栈地址传递过去了,new关键字大体就是在堆中开辟一个新空间,然后将空间地址存储到栈中,由于ref将栈地址传递过来了,所以就将栈中的存储地址替换为了新开辟的堆地址了,而不带ref的引用类型本身传递的只是堆中地址的引用,所以new关键字等于说将形参重新开辟空间和分配了,和实参已经不是同一个地方了。

总结

ref传递的是栈地址

不带ref传递的是栈中存储的值,可能是一个值,也可能是一个地址引用。

c#入参使用引用类型为啥要加ref?的更多相关文章

  1. 引用类型前需要加ref?

    方法的参数前加ref代表的是传的参数的地址,值类型前加ref的作用相当于把这个值类型当成引用类型在用,那引用类型作为参数有一种情况也需要加ref,不然得到的值会有差. 不加ref: class Pro ...

  2. js的replace函数入参为function时的疑问

    近期在写js导出excel文件时运用到replace方法,此处详细的记录下它各个参数所代表的的意义. 定义和用法 replace() 方法用于在字符串中用一些字符替换另一些字符,或替换一个与正则表达式 ...

  3. oracle常用函数以及调用入参为record的存储过程的方法,

    转自:http://www.cnblogs.com/zhangronghua/archive/2007/08/20/862812.html SQL中的单记录函数1.ASCII返回与指定的字符对应的十进 ...

  4. Egg.js 中入参的校验

    日常作业中免不了频繁处理 GET/POST 的入参,你当然可以每个 action 中都重复地去做这些事情, 从 query 或 body 取出入参, 对可选的入参进行判空, 处理入参的类型转换, 对入 ...

  5. Saiku根据入参日期查询出对应的数据(二十)

    Saiku根据入参日期查询出对应的数据 之前好像有写过一篇博客关于saiku date range的,现在进一步更新啦!!! 这里的日期筛选会更完善一些,需要提供两个参数 开始日期与结束日期(star ...

  6. 来啊踩fastjson打印入参导致业务跑偏的坑

    线上代码对日志的记录,重要性自不必说.但是怎样记录日志也是有讲究的! 日志可以直接在每个方法中进行日志记录,优点是想怎么记就怎么记,缺点是记日志的代码可能会超过你的业务代码,可读性急剧下降,这也是日志 ...

  7. Oracle存储过程入参传入List集合的小例子

    第一步:创建一个对象类型 create or replace type STUDENT as object( id ), name ), age ) ); / 第二步:创建一个数组类型 (任意选择下面 ...

  8. spring boot 三种入参

    先来讲述下最简单的使用get请求用户信息的实现方式,代码如下,写好后直接在Application类点击右键有个RunAs,点击后会自动运行,运行成功后可以使用http发包工具进行测试,这里推荐使用ch ...

  9. springboot-mvc:入参日期类型转换String->Date

    4种方式: 1.通过在application.ym中配置 spring.mvc.data-format: yyyy-MM-dd HH:mm:ss ,使用的是ParserConverter 优点:简单的 ...

  10. @Validated和@Valid区别:Spring validation验证框架对入参实体进行嵌套验证必须在相应属性(字段)加上@Valid而不是@Validated

    Spring Validation验证框架对参数的验证机制提供了@Validated(Spring's JSR-303规范,是标准JSR-303的一个变种),javax提供了@Valid(标准JSR- ...

随机推荐

  1. 如何实现Windows平台RTMP播放器/RTSP播放器播放窗口添加OSD文字叠加

    好多开发者在做Windows平台特别是单屏多画面显示时,希望像监控摄像机一样,可以在播放画面添加OSD台标,以实现字符叠加效果,大多开发者可很轻松的实现以上效果,针对此,本文以大牛直播SDK (Git ...

  2. python中使用数组作为索引

    链接:https://blog.csdn.net/yzlh2009/article/details/114118470 情况一,索引数组为整数值 情况二,索引数组为bool值

  3. SpringBoot源码学习1——SpringBoot自动装配源码解析+Spring如何处理配置类的

    系列文章目录和关于我 一丶什么是SpringBoot自动装配 SpringBoot通过SPI的机制,在我们程序员引入一些starter之后,扫描外部引用 jar 包中的META-INF/spring. ...

  4. 使用Elasticsearch的processors来对csv格式数据进行解析

    来源数据是一个csv文件,具体内容如下图所示: 导入数据到es中 有两种办法,第一种是在kibana界面直接上传文件导入 第二种方法是使用filebeat读取文件导入 这里采用第二种办法 配置文件名: ...

  5. Logstash:如何处理 Logstash pipeline 错误信息

    转载自:https://elasticstack.blog.csdn.net/article/details/114290663 在我们使用 Logstash 的时候经常会出现一些错误.比如当我们使用 ...

  6. 升级Gogs版本

    今天早上收到阿里云发的报警短信,大致内容如下: 前提分析: 公司代码代码仓库使用是Gogs搭建的,版本是0.11.34,二进制方式安装的,连接的是其他主机上的MySQL数据库,因此被检测到有这个漏洞 ...

  7. Zookeeper QuickStart

    环境版本 操作系统:CentOS release 6.6 (Final) java版本: jdk1.8 zookeeper版本: zookeeper-3.4.11 一. 安装jdk 此处省略 二. 安 ...

  8. SVM公式详尽推导,没有思维跳跃。

    假定数据集\(T=\{(x_1,y_1),(x_2,y_2),...,(x_n,y_n)\},x_n \in R_k, y_n \in \{1,-1\}\)线性可分,SVM的优化目标是: 优化一个超平 ...

  9. 洛谷P1884 [USACO12FEB]Overplanting S (矩形切割)

    一种矩形切割的做法: 1 #include<bits/stdc++.h> 2 using namespace std; 3 typedef long long LL; 4 const in ...

  10. Netty 学习(十):ChannelPipeline源码说明

    Netty 学习(十):ChannelPipeline源码说明 作者: Grey 原文地址: 博客园:Netty 学习(十):ChannelPipeline源码说明 CSDN:Netty 学习(十): ...