zoukankan      html  css  js  c++  java
  • C#中值传递与引用传递的区别

    以值传递参数

    当实参当作值来传递时,就产生了一个新的拷贝。

     class Test     {  

         static void Main(string[] args)         {      

                  int x=8;            

                    Fo(x);  

                Console.WriteLine("x={0}", x);  

           }  

             static void Fo(int p)  

           { 

               p = p + 1;  

               Console.WriteLine("p={0}", p);  

           }  

       }    程序运行结果为:p=9,x=8;即X的值不会受P影响,给P赋一个新值并不会改变X的内容,因为P和X存在于内存中不同的位置。

     

    同理,用传值的方式传递一个引用类型对象时,只是复制这个对象本身,即复制其地址值,而不是它指代的对象。下面代码中Fo中看到的StringBuilder对象,就是在Main方法中实例化的那一个,只是有不同的引用指向它而已。  static void Fo(StringBuilder foSB)  

    {       

          foSB.Append("test");  

          foSB = null;  

    }  

    static void Main()  

    {  

    StringBuilder sb = new StringBuilder();  

    Fo(sb);  

    Console.WriteLine(sb.ToString());  

    }    运行结果:test.

    换句话说,sb和foSB是指向同一对象的不同引用变量。因为foSB是引用的拷贝,把它置为null并没有把sb置为 null.

     

    ref修饰符

    当使用rel关键字时,表示是用引用的方式传递参数。实参和形参引用的都是同一个对象,改变其中一个的引用值,另一个也会改变。  

    static void Main(string[] args)  

    {  

               int x = 8;  

               Fo(ref x);  

               Console.WriteLine("x={0}", x);  

    }  

     

    static void Fo(ref int p)  

    {  

                

               p = p + 1;  

               Console.WriteLine("p={0}", p);  

    }    运行结果:P=9;X=9  。  

    如果在函数FO中改变P的值,则X的值也会随之改变。  static void Main(string[] args)  

    {  

                int x = 8;  

                Fo(ref x);  

                Console.WriteLine("x={0}", x);  

    }  

      

    static void Fo(ref int p)  

    {  

                p = 10;  

                p = p + 1;  

                Console.WriteLine("p={0}", p);  

    }  运行结果:P=11,X=11;

    ref修饰符在写函数和调用函数时都一定要出现。

    ref修饰符主要应该于实现交换的方法中。  static void Swap(ref string a, ref string b)  

    {  

                string temp = a;  

                a = b;  

                b = temp;  

    }  

    static void Main()  

    {  

                string x = "Hello";  

                string y = "World";  

                Swap(ref x, ref y);  

                Console .WriteLine (x);  

                Console.WriteLine(y);  

     运行结果:World Hello

     

    out修饰符

    out修饰符与ref修饰符非常相似,除了以下两点为:

    一,在调用函数时不需要赋值。

    二,在函数退出前必须赋值。

    out修饰符通常用于需要从方法中获取多个返回值的时候  static void Split(string name, out string firstName, out string lastName)  

    {  

                int i=name .LastIndexOf (' ');  

                firstName =name .Substring (0,i);  

                lastName =name .Substring (i+1);  

    }  

    static void Main()  

    {  

                string name = "Steven Francis";  

                string a, b;  

                Split(name, out a, out b);  

                Console .WriteLine (a);  

                Console .WriteLine (b);  

    }    运行结果:a=Steven,b=Francis  总结:  1,值传递时,为什么被调用的方法中的形参值的改变不会影响到相应的实参?

    答:因为按值传递时,系统首先为被调用的方法的形参分配内存空间,然后把实参中的值按位置一一对应“复制”给形参。形参中存储的值只是一份实参的拷贝,因此被调用方法中形参值的任何改变都不会影响到相应的形参。

      2,值传递和引用传递有什么不同,什么是值参数,它以什么方式传递? 答:值 传递时,系统首先为被调用方法的形参分配内存空间,并将实参的值按位置一一对应地复制给形参,此后,被调用方法中形参值得任何改变都不会影响到相应的实 参; 而引用传递时,系统不是将实参本身的值复制后传递给形参,而是将其引用值(即地址值)传递给形参,因此,形参所引用的该地址上的变量与传递的实参相同,方 法体内相应形参值得任何改变都将影响到作为引用传递的实参。

      3,什么是形参,什么是实参? 答: 形参:在定义函数中指定的参数就是形参,在未出现函数调用时,他们并不占内存中的存储单元,只有在发生函数调用时,函数中的形参才被分配内存单元。在调用结束后,形参所占的内存单元也被释放。
    实参:实参可以是常量、变量和表达式,但要求有确定的值。在调用时将实参的值赋给形参。在内存中,实参单元和形参单元是不同的单元。在调用函数时,给形参分配存储单元,并将实参对应的值传递给形参,调用结束后,形参单元被释放,实参单元仍保留原值。
    理解: 实参就是送进去方法中的东西~~行参就是把送进来的东西在方法中进行拷贝加工,加工完后方法就返回一个东西--返回值。

      值传递的时候,实参是不变的~形参是随着计算而变化的~~ 引用传递的时候~~行参怎么变~实参就怎么变.... 
    参数的传递分为:1.值方式参数传递,2.引用方式参数传递。

  • 相关阅读:
    禁止页面缓存 移动端
    常见的请求头与响应头介绍
    阮一峰 ES6学习
    禁止页面缩放 移动端
    10月30日学习日志
    11月7日学习日志
    11月6日学习日志
    10月31日学习日志
    11月1日学习日志
    11月9日学习日志
  • 原文地址:https://www.cnblogs.com/jiangyongyawen/p/4241193.html
Copyright © 2011-2022 走看看