C# 参数2:引用参数----值类型和引用类型

using System;

namespace ConsoleOnlyTest
{
    internal class Program
    {
        static void Main(string[] args)
        {
            //按引用传值类型时,不根据方法中的形参创建副本,而是直接指向实参的存储位置,所以后续修改都是对原实参的修改
            //因为未创建副本,所以在方法体内修改参数,会影响原实参的值
            Console.WriteLine("实验1:按引用传递值类型:\n");
            int a = 10;
            Console.WriteLine("主函数中调用方法1前:\n哈希代码:{0}\nx的值:{1}\n", a.GetHashCode(), a);
            Console.WriteLine("--------------------------");
            Method1(ref a);
            Console.WriteLine("主函数中调用方法1后:\n哈希代码:{0}\nx的值:{1}\n", a.GetHashCode(), a);
            Console.WriteLine("\n=====================================================================\n");

            //按引用传递引用类型且在方法中创建新实例时,不创建实参的副本,而是直接指向实参的存储位置,并在堆上创建新实例,形参实参都变为指向新实例,老实例会被GC
            //因为未创建副本,所以在方法体内修改参数,会影响原实参的值
            //可以观察下方实验,调用方法前后哈希代码产生变化,说明引用变量指向了新实例
            Console.WriteLine("实验2:按引用传递引用类型(在方法中新建实例):\n");
            Student outterStu1 = new Student();
            outterStu1.Name = "于小谦";
            Console.WriteLine("主函数中调用方法2前:\n哈希代码:{0}\n姓名:{1}\n", outterStu1.GetHashCode(), outterStu1.Name);
            Console.WriteLine("--------------------------");
            Method2(ref outterStu1);
            Console.WriteLine("主函数中调用方法2后:\n哈希代码:{0}\n姓名:{1}\n", outterStu1.GetHashCode(), outterStu1.Name);
            Console.WriteLine("\n=====================================================================\n");

            //按引用传递引用类型且未在方法中创建新实例时,不创建实参的副本,而是直接指向实参的存储位置,所以后续修改都是对原实参的修改
            //因为未创建副本,所以在方法体内修改参数,会影响原实参的值
            //可以观察下方实验,调用方法前后哈希代码未产生变化,说明引用变量仍指向原实例
            Console.WriteLine("实验3:按引用传递引用类型(不在方法中新建实例):\n");
            Student outterStu2 = new Student();
            outterStu2.Name = "于小谦";
            Console.WriteLine("主函数中调用方法3前:\n哈希代码:{0}\n姓名:{1}\n", outterStu2.GetHashCode(), outterStu2.Name);
            Console.WriteLine("--------------------------");
            Method3(ref outterStu2);
            Console.WriteLine("主函数中调用方法3后:\n哈希代码:{0}\n姓名:{1}\n", outterStu2.GetHashCode(), outterStu2.Name);
            //该方法和按值传递引用类型的结果相同(用不用ref结果相同),但内存原理不同,按值传递引用类型是创建了实参的副本同时指向原实例
            Console.WriteLine("\n=====================================================================\n");
        }

        public static void Method1(ref int a)
        {
            a += 10;
            Console.WriteLine("在方法1中:\n哈希代码:{0}\nx的值:{1}\n", a.GetHashCode(), a);
        }
        public static void Method2(ref Student stu)
        {
            stu = new Student() { Name = "于谦" };
            Console.WriteLine("在方法2中:\n哈希代码:{0}\n姓名:{1}\n", stu.GetHashCode(), stu.Name);
        }
        public static void Method3(ref Student stu)
        {
            stu.Name = "于谦";
            Console.WriteLine("在方法3中:\n哈希代码:{0}\n姓名:{1}\n", stu.GetHashCode(), stu.Name);
        }
    }
    class Student
    {
        public string Name { get; set; }
    }
}

        二、引用参数

        1、值类型:

        按引用传递值类型,方法调用时不建立实参的副本,而是直接指向实参的内存地址,方法内对变量的修改,均为修改实参。

        2、引用类型:

        按引用传递引用类型,分以下两种情况:

        ①在方法内创建了新实例。这种情况下不会创建实参的副本,而是直接指向实参的内存地址,方法内对变量的修改,均会修改实参。最终变量指向新实例,老实例会被回收。参考上述代码中方法2的哈希代码发生了改变。

        ②在方法体内未创建新实例。这种情况下与①相同的是也不会创建实参的副本,直接指向实参的内存地址,方法内对变量的修改,均会修改实参。但引用变量指向的实例不变,参考上述代码中方法3的哈希代码并未改变。

        注1:②中的结果与“按值传递引用类型”的结果一致,但两种情况的内存运行过程是不一样的。按值传递引用类型是形参在内存中创建了实参的副本,只不过副本与实参指向同一个实例。区别就在于一个创建了副本,一个未创建副本。

        注2:以上表述“未创建副本”,可以通过监视器中查看&outterStu和&stu的值来确认,在按引用传递时两者结果均相同。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值