C#篇(三)——函数传参之引用类型和值类型
首先应该认清楚在C#中只有两种类型:
1、引用类型(任何称为“类”的类型)
2、值类型(结构或枚举)
先来认识一下引用类型和值类型的区别:

函数传参之引用类型:
1、先来一个简单的引用类型传参的实例:
//使用了C#6.0的一个新特性:using static System.Console;
	class Program
    {
        static void StartTest1(string test)
        {
            test = "test2";
            WriteLine(test);//输出:"test2"
        }
        static void StartTest2(string test)
        {
            test = null;
            WriteLine(test);//输出:(空白)
        }
        static void Main(string[] args)
        {
            string test = "test1";
            WriteLine(test);//输出:"test1"
            StartTest1(test);
            WriteLine(test);//输出:"test1"
            StartTest2(test);
            WriteLine(test);//输出:"test1"
        }
    }
输出结果:
test1
test2
test1
test1
结果分析:
首先明白字符串(string)类型是引用类型,但改变了它的值之后,并没有影响到函数外面那个实参的值,这可能与大家的常识有点相违背,因为我们都知道若是变量以"引用传递"的方式传递,那么调用的方法可以通过更改其参数值,来改变调用者的变量值,但这里有一点需要说明的是:"引用传递"不是等价于引用类型传参,这是很多人的误解的地方。其实在C#当中,引用类型和值类型默认都是以“传值”的方式传递数值(引用)的(引用类型的值就是引用(类似索引或地址),而不是对象本身)。
请看下图详细分析:

2、再来一个略微复杂的引用类型传参的实例:
 	class Program
    {
        static void StartTest1(StringBuilder test)
        {
            test.Append("test2");
            WriteLine(test);//输出:"test1test2"
        }
        static void StartTest2(StringBuilder test)
        {
            test = null;
            WriteLine(test);//输出:(空白)
        }
        static void Main(string[] args)
        {
            StringBuilder test = new StringBuilder();
            test.Append("test1");
            WriteLine(test);//输出:"test1"
            StartTest1(test);
            WriteLine(test);//输出:"test1test2"
            StartTest2(test);
            WriteLine(test);//输出:"test1test2"
            ReadKey();
        }
    }
输出结果:
test1
test1test2
test1test2
test1test2
结果分析:
StringBuilder和string同样是引用类型,那为什么最终的StringBuilder类型值改变了呢?其实这里要纠正一下,真正改变的不是StringBuilder类型值(也就是引用的值),而是引用指向的字符数组引用指向的对象值改变了。在StringBuilder类里面封装了一个字符数组(最终的输出的就是这个字符数组,而那些操作也是对这个字符数组进行操作)。
结合上面两个实例,对于引用类型传参,从这里可以得出一个小结论:
1、在函数里面,若直接改变的是引用的值(也就是地址),那么之后的操作都不会影响到函数外面的那个变量
2、在函数里面,若直接改变的是引用指向的对象(值类型)的值(甚至更深层次的对象的值),那么就会影响到函数外面的变量
所以区分清楚改变的是引用的值还是引用指向的对象(值类型)的值是关键。
3、再来一个综合的引用类型传参的实例:
    class Program
    {
        class Test
        {
            public int index;//值类型
            public StringBuilder builder;//引用类型
            public string Result{
                get{return $"{index}:{builder.ToString()}";}
            }
        }
        static void StartTest(Test test)
        {
            test.index++;
            test.builder.Append("test2");
            WriteLine(test.Result);//输出:"2:test1test2"
            test.index = new int();
            test.builder = new StringBuilder();
            test.builder.Append("test3");
            WriteLine(test.Result);//输出:"0:test3"
        }
        static void Main(string[] args)
        {
            Test test = new Test {
                index = 0,
                builder = new StringBuilder()
            };
            test.index++;
            test.builder.Append("test1");
            WriteLine(test.Result);//输出:"1:test1"
            StartTest(test);
            WriteLine(test.Result);//输出:"0:test3"
        }
    }
输出结果:
1:test1
2:test1test2
0:test3
0:test3
结果分析:
略
[若是能够明白1和2中的分析,这个应该没有问题的]
函数传参之值类型:
简单的值类型传参这里就不演示了,来一个含有引用类型的值类型传参实例(只是将上例中的struct改为了class,这样好做对比):
    class Program
    {
        struct Test
        {
            public int index;//值类型
            public StringBuilder builder;//引用类型
            public string Result{
                get{return $"{index}:{builder.ToString()}";}
            }
        }
        static void StartTest(Test test)
        {
            test.index++;
            test.builder.Append("test2");
            WriteLine(test.Result);//输出:"2:test1test2"
            test.index = new int();
            test.builder = new StringBuilder();
            test.builder.Append("test3");
            WriteLine(test.Result);//输出:"0:test3"
        }
        static void Main(string[] args)
        {
            Test test = new Test {
                index = 0,
                builder = new StringBuilder()
            };
            test.index++;
            test.builder.Append("test1");
            WriteLine(test.Result);//输出:"1:test1"
            StartTest(test);
            WriteLine(test.Result);//输出:"1:test1test2"
        }
    }
输出结果:
1:test1
2:test1test2
0:test3
1:test1test2
结果分析:
首先应该明白,值类型以"传值"方式传递时,是一种浅拷贝,所以对于引用类型,只是复制了引用的值,副本(形参)中的引用指向的对象还是同一个。其他的自己分析应该明白。
结论:
1、无论是引用类型还是值类型,永远不会传递对象本身。涉及到一个引用类型时,要么以“引用传递”的方式(使用了ref或out关键字)传递变量,要么以“传值”的方式传递参数值(引用)。所以,通常函数传参(不论是引用类型还是值类型),都是以“传值”的方式传递的,只是要明白引用类型的值是引用本身(相当于一个索引或地址,而这个索引或地址最终指向的才是对象本身)。
2、“引用方式”传递与“传值”传递方式最大的区别就是“引用方式”要使用ref或out关键字修饰,所以以这个为标准去区分函数传参的方式(而不是以类型是引用类型还是值类型)。
3、对于传入函数的引用类型变量,最终会不会受到函数内部修改的影响,需要区分清楚函数内部改变的是引用的值还是引用指向的对象(值类型)的值。
C#篇(三)——函数传参之引用类型和值类型的更多相关文章
- 悉数 Python 函数传参的语法糖
		
TIOBE排行榜是程序开发语言的流行使用程度的有效指标,对世界范围内开发语言的走势具有重要参考意义.随着数据挖掘.机器学习和人工智能相关概念的风行,Python一举收获2018年年度语言,这也是Pyt ...
 - [Java]_函数传参的疑惑与思考
		
问题来源于leetcode上的两道题 Path Sum I && II,分别写了两个dfs. void dfs(TreeNode node , int sum , ArrayList& ...
 - python中给函数传参是传值还是传引用
		
首先还是应该科普下函数参数传递机制,传值和传引用是什么意思? 函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题.基本的参数传递机制有两种:值传递和引用传 ...
 - python函数传参是传值还是传引用?
		
首先还是应该科普下函数参数传递机制,传值和传引用是什么意思? 函数参数传递机制问题在本质上是调用函数(过程)和被调用函数(过程)在调用发生时进行通信的方法问题.基本的参数传递机制有两种:值传递和引用传 ...
 - pytest十一:函数传参和 firture 传参数 request
		
为了提高代码的复用性,我们在写用例的时候,会用到函数,然后不同的用例去调用这个函数.比如登录操作,大部分的用例都会先登录,那就需要把登录单独抽出来写个函数,其它用例全部的调用这个登录函数就行.但是登录 ...
 - 『Python × C++』函数传参机制学习以及对比
		
一.Python函数传参 在python中,函数传参实际上传入的是变量的别名,由于python内在的变量机制(名称和变量值相互独立),只要传入的变量不可变(tuple中的元素也要是不可变的才行),那么 ...
 - 函数传参传的是啥的思考【java Python】
		
今天看<java 核心 卷1>的时候,作者提到了函数传参的问题,他提到,java传参,传的是值,而不是引用,然后,函数将要传的实参的值(如果实参是基本数据类型,那么就是值.如果实参是对象, ...
 - JS——变量和函数的预解析、匿名函数、函数传参、return
		
JS解析过程分为两个阶段:编译阶段.执行阶段.在编译阶段会将函数function的声明和定义都提前,而将变量var的声明提前,并将var定义的变量赋值为undefined. 匿名函数: window. ...
 - pytest_函数传参和firture传参数request
		
前言为了提高代码的复用性,我们在写用例的时候,会用到函数,然后不同的用例去调用这个函数. 比如登录操作,大部分的用例都会先登录,那就需要把登录单独抽出来写个函数,其它用例全部的调用这个登陆函数就行. ...
 
随机推荐
- shadowOffset 具体解释
			
x向右为正,y向下为正 1.y<0 UILabel *label=[[UILabelalloc] initWithFrame:CGRectMake(40,40, 250,50)]; label. ...
 - HDFS HA架构以及源代码引导
			
HA体系架构 相关知识介绍 HDFS master/slave架构,HDFS节点分为NameNode节点和DataNode节点. NameNode存有HDFS的元数据:主要由FSImage和EditL ...
 - 杭电3501Calculation 2 欧拉函数
			
Calculation 2 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) To ...
 - BZOJ 1007: [HNOI2008]水平可见直线  平面直线
			
1007: [HNOI2008]水平可见直线 Description 在xoy直角坐标平面上有n条直线L1,L2,...Ln,若在y值为正无穷大处往下看,能见到Li的某个子线段,则称Li为可见的,否则 ...
 - systemd服务管理---systemctl命令列出所有服务
			
1.列出系统所有服务 #systemctl list-units --all --type=service
 - MFC补码原码反码转换工具
			
/*_TCHAR str[100] = { 0 }; wsprintf(str, _T("%d"),num);*/ ; CString str; m_edit1.GetWindow ...
 - Codeforces 987B. High School: Become Human
			
解题思路: 1.题意:判断x^y和y^x谁大谁小. 2.由于x^y和y^x太大了,时间复杂度也不允许,所以做同等变换,比较e^(ylnx)和e^(xlny). 3.即为比较ylnx和xlny的大小. ...
 - jquery 几种类选择器方式
			
代码如下: <%@ Page Language="C#" AutoEventWireup="true" CodeBehind="TestClas ...
 - Jquery 设置class 和 div CSS
			
Jquery 设置class 和 div CSS 1 Jquery 根据标签内容获取标签div,从而修改该div CLASS //追加 $('label:contains("labelcon ...
 - 【原创】JMS发布者订阅者【异步接收消息】
			
发布订阅模式和PTP方式不同之处为后者依赖于一个Topic话题: package com.thunisoft.jms.mine.topic; import java.util.HashMap; imp ...