值传递与引用传递

JVM在执行方法main时,会创建java栈(属于某一个线程,不共享),并将变量 a存储到局部变量区,在调用方法change时,根据(invoke**)指令,为change创建一个新的栈帧,并将参数保存在新栈帧的局部变量区,在执行ireturn指令后,将栈顶元素返回到调用方法的栈中,创建的栈帧也被撤销。PC寄存器指令恢复调用栈的下一条命令地址。继续执行--

首先要说明的是java中是没有指针的,java中只存在值传递,只存在值传递!!!  然而我们经常看到对于对象(数组,类,接口)的传递似乎有点像引用传递,可以改变对象中某个属性的值。但是不要被这个假象所蒙蔽,实际上这个传入函数的值是对象引用的拷贝,即传递的是引用的地址值,所以还是按值传递。

 值传递

    示例:

复制代码
public class Test3 {
    public static void change(int a){
        a=50;
    }
    public static void main(String[] args) {
        int a=10;
        System.out.println(a);
        change(a);
        System.out.println(a);
    }
}
复制代码

 很显然输出的 是10,10。传递的是值得一份拷贝,这份拷贝与原来的值没什么关系。

内存分析:

              

  引用传递

      示例:

复制代码
public class Test3 {
    public static void change(int []a){
        a[0]=50;
    }
    public static void main(String[] args) {
        int []a={10,20};
        System.out.println(a[0]);
        change(a);
        System.out.println(a[0]);
    }
}
复制代码

显然输出结果为10   50。实际传递的是引用的地址值。

内存分析:

            

 示例:

复制代码
class Emp {
    public int age;
}
public class Test {
    public static void change(Emp emp)
    {
        emp.age = 50;
        emp = new Emp();//再创建一个对象
        emp.age=100;
    }
    
    public static void main(String[] args) {
        Emp emp = new Emp();
        emp.age = 100;
        System.out.println(emp.age);
        change(emp);
        System.out.println(emp.age);
        System.out.println(emp.age);
    }
}
复制代码

输出为:100  50  50.

内存分析:

                

对于String类:

复制代码
public class Test {
    public static void change(String s){
        s="zhangsan";
    }
    
    public static void main(String[] args) {
        String s=new String("lisi");
        System.out.println(s);
        change(s);
        System.out.println(s);
    }
}
复制代码

输出为:lisi   lisi,由于String类是final修饰的,不可变,它会在内存中在开辟一块新空间。

原文地址:https://www.cnblogs.com/y654063871/p/6916507.html