论坛首页 入门技术论坛

Java中 引用调用 VS 按值调用

浏览 3167 次
精华帖 (0) :: 良好帖 (0) :: 新手帖 (0) :: 隐藏帖 (0)
作者 正文
   发表时间:2007-11-17  

               貌似大多数的初学者对java中函数的调用是按照"引用调用"还是"按值调用"存在一定的误区。这些调用方式,表述了编

程语言如何管理向函数(function)或方法传入的参数,归根到底,是要理解java如何管理参数的传递。

              按值调用意味着被调用的函数在幕后对参数进行了拷贝,函数中的代码操作的是这个拷贝,意味着对参数进行的任何改

动,都会在函数执行完毕后被丢弃掉,原因是你所做的改动只作用于局部的拷贝,而非传入的参数,参数的拷贝只在方法范围

内有效,当超出方法的执行范围后,那局部的拷贝的作用域也只限于方法的范围内。

            按引用调用意味着函数操作的和传入的参数,在物理上是相同的,由于两个引用指向的是堆中的同一个对象,对参数任

何改动都会被保存下来。


            需要说明的是java中对基本类型的操作是按值调用的方式进行的,而对引用类型的操作是按照引用调

用的方式进行的,下面的代码显示了java中对基本类型的按值调用的方式的处理结果。

java 代码
  1. package cn.com.two;   
  2.   
  3. public class TestCallByValue {   
  4.   
  5.        
  6.     public static void main(String[] args) {   
  7.            
  8.         int count=1;   
  9.         increment(count);   
  10.         System.out.println("count=="+count);   
  11.            
  12.     }   
  13.        
  14.     public static void increment(int count){   
  15.         count++;   
  16.     }   
  17.   
  18. }   

执行结果:

java 代码
  1. count==1  


             在这里increment方法改变了count的值,这个改变只作用于局部的拷贝,而这个局部的拷贝在方法执行后就消失了,最

后count保存的还是原来的值。


           下面再来看看引用调用的方式,先回顾一下,引用是一个指针,---指向的是内存中的一个地址,而该方法中参数只是创

建了一个新的指针指向同样的内存位置,如果你在调用的方法的内部,将一个新的内存地址(也就是一个不同的对象)赋值给这个

引用,新的地址在方法执行完后同样会被丢弃。对引用方法的调用导致该引用所指向的对象的状态的永久性的改变。

java 代码
  1. package cn.com.two;   
  2.   
  3. public class TestCallByReferences {   
  4.   
  5.     /**  
  6.      * @param args  
  7.      */  
  8.     public static void main(String[] args) {   
  9.            
  10.         Customer customer=new Customer(1);   
  11.         Customer customer_2=increment(customer);   
  12.         System.out.println("Customer id=="+customer.getId());   
  13.         System.out.println("Customer_2 id="+customer_2.getId());   
  14.   
  15.     }   
  16.        
  17.     public static Customer increment(Customer customer){   
  18.         customer.setId(customer.getId()+1);   
  19.         customer=new Customer(22);   
  20.         return customer;   
  21.     }   
  22.   
  23. }   
  24.   
  25. class Customer{   
  26.        
  27.      private int id;   
  28.         
  29.      public Customer(int id){   
  30.          this.id=id;   
  31.      }   
  32.   
  33.     public int getId() {   
  34.         return id;   
  35.     }   
  36.   
  37.     public void setId(int id) {   
  38.         this.id = id;   
  39.     }   
  40.         
  41.         
  42. }  


          在这里创建了一个ID为1的客户,调用increment方法,传如的是一个引用,指向了堆中创建的一个Customer对象,对这

个引用执行的操作将直接反应在堆的对象上,接着创建了一个新的Customer对象,将它的地址赋给customer参数引用,然后

返回customer引用,此时它指向的是堆中ID为22的那个Customer对象,最后打印结果如下:

java 代码
  1. Customer id==2  
  2. Customer_2 id=22  
   发表时间:2007-11-20  
java里的所有基本类型都是按值进行传递的,所有对象都是按引用进行传递的,不过有一个例外就是String对象,由于它是终态的,所有虽然是按引用传递,但是修改的时候是生成一个新的字符串的!
0 请登录后投票
   发表时间:2007-11-21  
String对象并不例外,这只是他的特性,你可以自己写一个不可改变的对象来成为你所谓的例外的对象。
0 请登录后投票
   发表时间:2007-11-27  
cnng007 写道
String对象并不例外,这只是他的特性,你可以自己写一个不可改变的对象来成为你所谓的例外的对象。


我所说的例外就是因为java在设计String对象时所设置的特性,当然我说它是终态并不能正确的解释这个问题,因为终态是不允许进行修改的,应该是线程安全的。总之就是String虽然是引用传递,但是任何修改都导致新字符串的生成!
0 请登录后投票
论坛首页 入门技术版

跳转论坛:
Global site tag (gtag.js) - Google Analytics