zoukankan      html  css  js  c++  java
  • Java中的值传递和引用传递

    Java中,String类型和包装类型作为参数传递时,是属于值传递还是引用传递呢?

    Java中的值传递和引用传递

    Java中没有指针,所以也没有引用传递了,仅仅有值传递不过可以通过对象的方式来实现引用传递 类似java没有多继承 但可以用多次implements 接口实现多继承的功能 
    值传递:方法调用时,实际参数把它的值传递给对应的形式参数,方法执行中形式参数值的改变不影响实际参数的值。
    引用传递:也称为传地址。方法调用时,实际参数的引用(地址,而不是参数的值)被传递给方法中相对应的形式参数,在方法执行中,对形式参数的操作实际上就是对实际参数的操作,方法执行中形式参数值的改变将会影响实际参数的值。 
    Java参数按值传递
    面试题:当一个对象被当作参数传递到一个方法后,此方法可改变这个对象的属性,并可返回变化后的结果,那么这里到底是值传递还是引用传递? 
        答:是值传递。Java 编程语言只有值传递参数。当一个对象实例作为一个参数被传递到方法中时,参数的值就是该对象的引用一个副本。指向同一个对象,对象的内容可以在被调用的方法中改变,但对象的引用(不是引用的副本)是永远不会改变的。 
    ------------------------------------------------------------- 
    在 Java 应用程序中永远不会传递对象,而只传递对象引用。因此是按引用传递对象。但重要的是要区分参数是如何传递的,这才是该节选的意图。Java 应用程序按引用传递对象这一事实并不意味着 Java 应用程序按引用传递参数。参数可以是对象引用,而 Java 应用程序是按值传递对象引用的。 

    Java 应用程序中的变量可以为以下两种类型之一:引用类型或基本类型。当作为参数传递给一个方法时,处理这两种类型的方式是相同的。两种类型都是按值传递的;没有一种按引用传递。 

    按值传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的一个副本。因此,如果函数修改了该参数,仅改变副本,而原始值保持不变。按引用传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的内存地址,而不是值的副本。因此,如果函数修改了该参数的值,调用代码中的原始值也随之改变。如果函数修改了该参数的地址,调用代码中的原始值不会改变. 
    当传递给函数的参数不是引用时,传递的都是该值的一个副本(按值传递)。区别在于引用。在 C++ 中当传递给函数的参数是引用时,您传递的就是这个引用,或者内存地址(按引用传递)。在 Java 应用程序中,当对象引用是传递给方法的一个参数时,您传递的是该引用的一个副本(按值传递),而不是引用本身。

    Java 应用程序按值传递参数(引用类型或基本类型),其实都是传递他们的一份拷贝.而不是数据本身.(不是像 C++ 中那样对原始值进行操作。)

    例1:

    //在函数中传递基本数据类型,  
    public class Test {     
    
         public static void change(int i, int j) { 
             int temp = i;     
             i = j;     
             j = temp;     
         }     
        public static void main(String[] args) {
            int a = 3;     
            int b = 4;     
            change(a, b); 
            System.out.println("a=" + a);
            System.out.println("b=" + b);
        }     
     }   
        
    结果为:     
     a=3    
     b=4 
    原因就是 参数中传递的是 基本类型 a 和 b 的拷贝,在函数中交换的也是那份拷贝的值 而不是数据本身; 

    例2:

    //传的是引用数据类型     
     public class Test { 
       {
        public static void change(int[] counts) { 
            int[] count = { 1, 2, 3, 4, 5 };     
         System.out.println("before change:"+count[0]);
    	change(count);
    	System.out.println("after change:"+count[0]);
        }     
        
       public static void main(String[] args) {     
           int[] count = { 1, 2, 3, 4, 5 };     
            change(count);     
        }     
     } 
    结果:
      before change:1   6   after change:6
    在方法中 传递引用数据类型int数组,实际上传递的是其引用count的拷贝,他们都指向数组对象,在方法中可以改变数组对象的内容。即:对复制的引用所调用的方法更改的是同一个对象。

    例3:

    //对象的引用(不是引用的副本)是永远不会改变的   
     class A { 
         int i = 0;     
     } 
         
    public class Test {     
       
     public static void add(A a) { 
           a = new A();    
           a.i++;  
      }   
    
         public static void main(String args[]) {     
             A a = new A();     
             add(a);   
             System.out.println(a.i);     
         }     
     }
    结果:
    before change:0
    after change:0    
      
    在该程序中,对象的引用指向的是A ,而在change方法中,传递的引用的一份副本则指向了一个新的OBJECT,并对其进行操作。   
    而原来的A对象并没有发生任何变化。 引用指向的是还是原来的A对象。 

     例3: String 不改变,数组改变 

     public class Example { 
     String str = new String("good"); 
     
    char[] ch = { 'a', 'b', 'c' }; 
    
    public static void main(String args[]) { 
    Example ex = new Example(); 
    System.out.println("before change: " +ex.str +" and "+ new String(ex.ch));
    ex.change(ex.str, ex.ch); 
    System.out.println("after change: " +ex.str +" and "+ new String(ex.ch));
    } 
    
    public void change(String str, char ch[]) { 
     str = "test ok"; 
     ch[0] = 'g'; 
    } 
    } 
    程序3输出的是 
    before change: good and abc
    after change: good and gbc.

    例4:

    class Test  
     { 
       public static void main(String args[]) 
       {  
            int val;
            StringBuffer sb1, sb2;
    	val = 10;
    	sb1 = new StringBuffer("apples");
    	sb2 = new StringBuffer("pears");
    	System.out.println("before modify: val is " + val+ ", sb1 is " + sb1+ ", sb2 is " + sb2);
    	// 按值传递所有参数
    	modify(val, sb1, sb2);
    	System.out.println("after modify: "+ " val is " + val +", sb1 is " + sb1 + ", sb2 is " + sb2);
       }   
    
       public static void modify(int a, StringBuffer r1,  
                                 StringBuffer r2) 
      {  
           a = 0;
           r1 = null; //1
           r2.append(" taste good"); 
      } 
    }  
    结果:  
    before modify: val is 10, sb1 is apples, sb2 is pears
    after modify:  val is 10, sb1 is apples, sb2 is pears taste good 

    这段代码声明了三个变量:一个整型变量和两个对象引用。设置了每个变量的初始值并将它们打印出来。然后将所有三个变量作为参数传递给 modify 方法。
    modify 方法更改了所有三个参数的值: 
    将第一个参数(整数)设置为 0。 
    将第一个对象引用 r1 设置为 null。 
    保留第二个引用 r2 的值,但通过调用 append 方法更改它所引用的对象(这与前面的 C++ 示例中对指针 p 的处理类似)。 

    当执行返回到 main 时,再次打印出这三个参数的值。正如预期的那样,整型的 val 没有改变。对象引用 sb1 也没有改变。如果 sb1 是按引用传递的,正如许多人声称的那样,它将为 null。但是,因为 Java 编程语言按值传递所有参数,所以是将 sb1 的引用的一个副本传递给了 modify 方法。当 modify 方法在 //1 位置将 r1 设置为 null 时,它只是对 sb1 的引用的一个副本进行了该操作,而不是像 C++ 中那样对原始值进行操作。 
    另外请注意,第二个对象引用 sb2 打印出的是在 modify 方法中设置的新字符串。即使 modify 中的变量 r2 只是引用 sb2 的一个副本,但它们指向同一个对象。因此,对复制的引用所调用的方法更改的是同一个对象。 


    传值---传递基本数据类型参数 

    public class PassValue{ 
            static void exchange(int a, int b){//静态方法,交换a,b的值  
                int temp; 
                temp = a;  
                a = b; 
                b = temp;  
            } 
            public static void main(String[] args){  
               int i = 10;  
               int j = 100;  
               System.out.println("before call: " + "i=" + i + "\t" + "j = " + j);//调用前  
                exchange(i, j);                                                                    //值传递,main方法只能调用静态方法  
                System.out.println("after call: " + "i=" + i + "\t" + "j = " + j);//调用后  
            } 
        } 

        运行结果:  
                before call: i = 10        j = 100  
                after    call: i = 10        j = 100  
    说明:调用exchange(i, j)时,实际参数i,j分别把值传递给相应的形式参数a,b,在执行方法exchange()时,形式参数a,b的值的改变不影响实际参数i和j的值,i和j的值在调用前后并没改变。 

    引用传递---对象作为参数 
    如果在方法中把对象(或数组)作为参数,方法调用时,参数传递的是对象的引用(地址),即在方法调用时,实际参数把对对象的引用(地址)传递给形式参数。这是实际参数与形式参数指向同一个地址,即同一个对象(数组),方法执行时,对形式参数的改变实际上就是对实际参数的改变,这个结果在调用结束后被保留了下来。 

    class Book{  
            String name;  
            private folat price;  
            Book(String n,    float ){                //构造方法  
                name = n;  
                price = p;  
            }  
            static  void  change(Book a_book,    String n,    float p){    //静态方法,对象作为参数  
                    a_book.name = n; 
                    a_book.price = p;  
            }  
            public void output(){        //实例方法,输出对象信息 
                System.out.println("name: " + name + "\t" + "price: " + price);  
            }  
        } 
         public class PassAddr{  
            public static void main(String [] args){  
                Book b = new Book("java2",    32.5f);  
                System.out.print("before call:\t");        //调用前  
                b.output(); 
                b.change(b,    "c++",    45.5f);            //引用传递,传递对象b的引用,修改对象b的值  
                System.out.print("after call:\t");            //调用后  
                b.output();
            }  
        }  
        运行结果:  
                before    call:    name:java2        price:32.5 
                after       call:    name:c++          price:45.5 

    说明:调用change(b,"c++",45.5f)时,对象b作为实际参数,把引用传递给相应的形式参数a_book,实际上a_book也指向同一个对象,即该对象有两个引用名:b和a_book。在执行方法change()时,对形式参数a_book操作就是对实际参数b的操作。

    Java参数,不管是原始类型还是引用类型,传递的都是副本(有另外一种说法是传值,但是说传副本更好理解吧,传值通常是相对传址而言)。
        如果参数类型是原始类型,那么传过来的就是这个参数的一个副本,也就是这个原始参数的值,这个跟之前所谈的传值是一样的。如果在函数中改变了副本的值不会改变原始的值.
        如果参数类型是引用类型,那么传过来的就是这个引用参数的副本,这个副本存放的是参数的地址。如果在函数中没有改变这个副本的地址,而是改变了地址中的值,那么在函数内的改变会影响到传入的参数。如果在函数中改变了副本的地址,如new一个,那么副本就指向了一个新的地址,此时传入的参数还是指向原来的地址,所以不会改变参数的值。 

    ( 对象包括对象引用即地址和对象的内容) 
    a.传递值的数据类型:八种基本数据类型和String(这样理解可以,但是事实上String也是传递的地址,只是string对象和其他对象是不同的,string对象是不能被改变的,内容改变就会产生新对象。那么StringBuffer就可以了,但只是改变其内容。不能改变外部变量所指向的内存地址)。 
    b.传递地址值的数据类型:除String以外的所有复合数据类型,包括数组、类和接口.

  • 相关阅读:
    大数据之路Week08_day02 (Flume架构介绍和安装)
    Hive调优
    hive的shell用法(脑子糊涂了,对着脚本第一行是 #!/bin/sh 疯狂执行hive -f 结果报错)
    Week08_day01 (Hive 自定义函数 UDF 一个输入,一个输出(最常用))
    Week08_day01 (Hive开窗函数 row_number()的使用 (求出所有薪水前两名的部门))
    Week08_day01 (Hive实现按照指定格式输出每七天的消费平均数)
    Week08_day01 (Hive实现WordCount计数)
    SQL中 count(*)和count(1)的对比,区别
    大数据之路week07--day07 (修改mysql默认编码)
    10进制转换成16进制的函数(自写函数模板)
  • 原文地址:https://www.cnblogs.com/duanxz/p/2806733.html
Copyright © 2011-2022 走看看