貌似大多数的初学者对java中函数的调用是按照"引用调用"还是"按值调用"存在一定的误区。这些调用方式,表述了编
程语言如何管理向函数(function)或方法传入的参数,归根到底,是要理解java如何管理参数的传递。
按值调用意味着被调用的函数在幕后对参数进行了拷贝,函数中的代码操作的是这个拷贝,意味着对参数进行的任何改
动,都会在函数执行完毕后被丢弃掉,原因是你所做的改动只作用于局部的拷贝,而非传入的参数,参数的拷贝只在方法范围
内有效,当超出方法的执行范围后,那局部的拷贝的作用域也只限于方法的范围内。
按引用调用意味着函数操作的和传入的参数,在物理上是相同的,由于两个引用指向的是堆中的同一个对象,对参数任
何改动都会被保存下来。
需要说明的是java中对基本类型的操作是按值调用的方式进行的,而对引用类型的操作是按照引用调
用的方式进行的,下面的代码显示了java中对基本类型的按值调用的方式的处理结果。
- package cn.com.two;
- public class TestCallByValue {
- public static void main(String[] args) {
- int count=1;
- increment(count);
- System.out.println("count=="+count);
- }
- public static void increment(int count){
- count++;
- }
- }
执行结果:
- count==1
在这里increment方法改变了count的值,这个改变只作用于局部的拷贝,而这个局部的拷贝在方法执行后就消失了,最
后count保存的还是原来的值。
下面再来看看引用调用的方式,先回顾一下,引用是一个指针,---指向的是内存中的一个地址,而该方法中参数只是创
建了一个新的指针指向同样的内存位置,如果你在调用的方法的内部,将一个新的内存地址(也就是一个不同的对象)赋值给这个
引用,新的地址在方法执行完后同样会被丢弃。对引用方法的调用导致该引用所指向的对象的状态的永久性的改变。
- package cn.com.two;
- public class TestCallByReferences {
- /**
- * @param args
- */
- public static void main(String[] args) {
- Customer customer=new Customer(1);
- Customer customer_2=increment(customer);
- System.out.println("Customer id=="+customer.getId());
- System.out.println("Customer_2 id="+customer_2.getId());
- }
- public static Customer increment(Customer customer){
- customer.setId(customer.getId()+1);
- customer=new Customer(22);
- return customer;
- }
- }
- class Customer{
- private int id;
- public Customer(int id){
- this.id=id;
- }
- public int getId() {
- return id;
- }
- public void setId(int id) {
- this.id = id;
- }
- }
在这里创建了一个ID为1的客户,调用increment方法,传如的是一个引用,指向了堆中创建的一个Customer对象,对这
个引用执行的操作将直接反应在堆的对象上,接着创建了一个新的Customer对象,将它的地址赋给customer参数引用,然后
返回customer引用,此时它指向的是堆中ID为22的那个Customer对象,最后打印结果如下:
- Customer id==2
- Customer_2 id=22