引用类型

在Java中,引用类型的变量非常类似于C/C++的指针。引用类型指向一个对象,指向对象的变量是引用变量。

对象、数组都是引用数据类型。


Java中形参如果为基本类型的话,传递的是值,所以改变的只是形参,实参不会改变。(因为在内存中它指向的数据的地址没改变,所以不会发生改变)。

而传递对象或者数组的时候传递的是引用,改变了内存的堆中的数据,所以形参会改变实参


如果形参是基本类型的变量,则实参不会变(值传递)。

如果是形参是引用类型的变量(对象或数组),那么修改形参时,实参会跟着改变(引用传递)。


 

1 传入的形参是基本类型的变量

public class Java{
    public static void main(String[] args){
       int b =5;
       changeInt(b);
       System.out.println(b);
    }

    public static int changeInt(int a){
       a =10;
       return a;
    }
}

changeInt(b)之后输出的是5。

为什么把b传到changeInt()方法中之后,b的值没有被修改为10?

因为,基本类型和String均为值传递,想要实现你的需求只能改用对象,当然也可以使用集合类比如Map,或者数组(这个比较方便)。

也就是说,传入参数为字符串变量时同样如此。

说明:在java类执行时,基本类型变量是存储在栈中的,压栈顺序:(1)主方法压栈->(2)变量b压栈->(3)静态方法changeInt压栈->(4)压栈参数并修改为10->(5)退出changeInt方法,并弹出changeInt方法及其作用域内的数据->(6) 退出main方法。

打印结果5是打印经过方法修改为10后,退出方法后打印的变量的值,由于基本数据类型存储在栈中,因此退出方法后,变量的值仍然是变量最初所在栈位置的值5.


2 传入的形参是对象

public class Java{
    public static void main(String[] args)
    {
       int b =5;
       b = changeInt(b);
       System.out.println(b);
    }

    public static int changeInt(int a){
       a = 10;
       return a;
    }
}

changeInt(b)之后输出的是10。

3 传入的形参是数组

public class Java{

    public static void main(String[] args){

                   int[]a = {23,57};
                   print(a);
                   test(a);
                   print(a);
    }

    public static void test(int[] num){ //颠倒数组
        inttemp = 0;
        for(inti = 0;i < num.length/2; i++){
            temp= num[i];
            num[i]= num[num.length - 1 - i];
            num[num.length- 1 - i] = temp;
        }        
    }

    public static void print(int[] num){ //打印数组

        for(inti = 0; i < num.length; i++){
            System.out.print(num[i]+"");
        }           
        System.out.println();
    }
}

输出:

 23 57

 57 23

为什么在这里方法退出后,变量的值变不回去了呢?

说明:在java中,数组是引用类型变量,可以看成是一个对象,在创建一个数组时,和创建一个新的对象时同样的道理(毕竟创建时使用new关键字了),这个对象存放在堆内存中,栈内存中只是存放了它的位置索引(即在堆内存中的位置,相当于栈中放了它的门牌号),这样在test方法中引用了数组变量,同样是通过栈中存放的位置索引找到这个变量在堆内存中的位置,然后进行修改操作,test方法可不是将它重新压到栈中再修改的,是直接在堆中改,直接在堆中改,直接在堆中改(重要的事情说三遍),然后很淡定的退出了test方法,它是退出没事了,堆中的数组变量已经被不可逆转的修改了,因此test方法前和方法后打印出来的数组内容是不一样的。