Java中 引用调用 VS 按值调用 <转>

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

              按值调用意味着被调用的函数在幕后对参数进行了拷贝,函数中的代码操作的是这个拷贝,意味着对参数进行的任何改动,都会在函数执行完毕后被丢弃掉,原因是你所做的改动只作用于局部的拷贝,而非传入的参数,参数的拷贝只在方法范围内有效,当超出方法的执行范围后,那局部的拷贝的作用域也只限于方法的范围内。

           按引用调用意味着函数操作的和传入的参数,在物理上是相同的,由于两个引用指向的是堆中的同一个对象,对参数任何改动都会被保存下来。

           需要说明的是java中对基本类型的操作是按值调用的方式进行的,而对引用类型的操作是按照引用调用的方式进行的,下面的代码显示了java中对基本类型的按值调用的方式的处理结果。

packagecn.com.two;    
    public 
    classTestCallByValue {    
    public 
    static 
    voidmain(String[] args) { 
   
 
   

         
    intcount=1; 
   
 
   

                  increment(count); 
   
 
   

                  System.out.println("count=="+count); 
   
 
   

              } 
   
 
   

         
    public 
    static 
    voidincrement( 
    intcount){    
             count++; 
   
 
   

              } 
   
 
   

         }

执行结果:

java 代码

count==1 

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

          下面再来看看引用调用的方式,先回顾一下,引用是一个指针,---指向的是内存中的一个地址,而该方法中参数只是创建了一个新的指针指向同样的内存位置,如果你在调用的方法的内部,将一个新的内存地址(也就是一个不同的对象)赋值给这个引用,新的地址在方法执行完后同样会被丢弃。对引用方法的调用导致该引用所指向的对象的状态的永久性的改变。

java 代码

packagecn.com.two; 
   
 
   

         
    public 
    classTestCallByReferences {   
   
 
   

           /**       * @param args       */   
   
 
   
 
    public 
    static 
    voidmain(String[] args) {  
   
 
   

                 Customer customer= 
    newCustomer(1); 
   
 
   

                  Customer customer_2=increment(customer);  
   
 
   

                 System.out.println("Customer id=="+customer.getId());  
   
 
   

                 System.out.println("Customer_2 id="+customer_2.getId()); 
   
 
   

              }    
    public 
    staticCustomer increment(Customer customer){    
             customer.setId(customer.getId()+1); 
   
 
   

                  customer= 
    newCustomer(22); 
   
 
   

         
    returncustomer; 
   
 
   

              } 
   
 
   

         }    
   
 
   
 
    classCustomer{ 
   
 
   

         
    private 
    intid; 
   
 
   

         
    publicCustomer( 
    intid){    
    this.id=id; 
   
 
   

               } 
   
 
   

         
    public 
    intgetId() { 
   
 
   

         
    returnid; 
   
 
   

              } 
   
 
   

         
    public 
    voidsetId( 
    intid) { 
   
 
   

         
    this.id = id;    
         } 
   
 
   

         }

          在这里创建了一个ID为1的客户,调用increment方法,传如的是一个引用,指向了堆中创建的一个Customer对象,对这个引用执行的操作将直接反应在堆的对象上,接着创建了一个新的Customer对象,将它的地址赋给customer参数引用,然后返回customer引用,此时它指向的是堆中ID为22的那个Customer对象,最后打印结果如下:

java 代码

Customer id==2   Customer_2 id=22