通过底层原理理解Java是值传递还是引用传递?

本文学习目标或者巩固的知识点

  • 参数传递方式
    • 值传递
    • 引用传递
    • 指针传递
  • 彻底理解Java的值传递和引用传递
    • 从底层的角度分析
    • 值传递会发生复制行为

Java的参数传递例子

快手的一面面试曾经问到过此类题目,所以记下此篇加深印象。
问:求下面main方法中的输出结果,如果知晓结果则可跳过此篇

public static void main(String[] args) {
    int num = 1;
    值传递_基本类型(num);
    System.out.printf("值传递_基本类型方法结束后=>num=%d \n", num);
    Person person = new Person();
    person.name = "张三";
    引用传递_引用类型(person);
    System.out.printf("引用传递_引用类型方法结束后=>person.name=%s \n", person.name);
}

public static void 值传递_基本类型(int num){
    num = 2;
}
public static void 引用传递_引用类型(Person person){
    person.name = "李四";
}

public static class Person{
    public String name;
}

输出

值传递_基本类型方法结束后=>num=1 
引用传递_引用类型方法结束后=>person.name=李四 

什么是值传递?

百度百科: 值传递是指在调用函数时将 实际参数 复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数。

Java中基本数据类型:整型(4种)、浮点型(2种)、字符型(1种)、布尔型(1种)。

在赋值操作符(=)的作用下,将变量的值从调用者的作用域传递到被调用的作用域。

对于基本数据类型,复制的是值本身,而不是变量的内存地址。因此,如果一个变量在调用过程中被修改,那么在返回调用者的作用域时,其值仍然是复制时的值。

public static void main(String[] args) {
    int num = 1;
    Integer number = 1;
    char ch = '1';
    Character character = '1';
    值传递_基本类型(num,number,ch,character);
    System.out.printf("值传递_基本类型方法结束后=>num=%d, number=%d, ch=%c, character=%c \n", num, number, ch, character);
}
public static void 值传递_基本类型(int num, Integer number, char ch, Character character){
    num = 2;
    number = 2;
    ch = '2';
    character = '2';
    System.out.printf("值传递_基本类型方法内=>num=%d, number=%d, ch=%c, character=%c \n", num, number, ch, character);
}

值传递_基本类型方法内=>num=2, number=2, ch=2, character=2 
值传递_基本类型方法结束后=>num=1, number=1, ch=1, character=1 

什么是引用传递?

百度百科:所谓 引用传递 是指在调用函数时将实际参数的地址传递到函数中,那么在函数中对参数所进行的修改,将影响到实际参数。

在Java中与引用传递相类似的有引用类型的传递(如数组、Object类),在赋值操作符(=)的作用下,将变量的引用指向的对象从调用者的作用域传递到被调用的作用域

对于引用类型,复制的是对象的引用,而不是对象本身。因此,如果一个对象在调用过程中被修改,那么在返回调用者的作用域时,所引用的对象仍然是调用时的对象,但其内容可能会发生变化。比如:如果一个对象包含其他对象作为成员,那么在值传递过程中,只有对象的引用会被复制,而成员对象则不会被复制。这意味着在调用过程中,如果修改了对象内部的成员对象,那么在返回调用者的作用域时,该成员对象可能已经不再是最新的。

值传递:引用数据类型

public static void main(String[] args) {
    int[] arr = new int[1];
    char[] chs = new char[1];
    List<Integer> ls = new ArrayList<>();
    Person person = new Person();
    
    arr[0] = 1;
    chs[0] = '1';
    ls.add(1);
    person.name = "张三";
    
    System.out.printf("引用传递_引用类型方法执行前=> " +
            "\n arr[0]=%d, chs[0]=%c,ls.get(0)=%d, person.name=%s \n", arr[0], chs[0], ls.get(0), person.name);
    
    引用传递_引用类型(arr,chs,ls,person);
    
    System.out.printf("引用传递_引用类型方法结束后=> " +
            "\n arr[0]=%d, chs[0]=%c,ls.get(0)=%d, person.name=%s \n", arr[0], chs[0], ls.get(0), person.name);
}
public static void 引用传递_引用类型(int[] arr, char[] chs, List<Integer> ls, Person personCopy){
    arr[0] = 2;
    chs[0] = '2';
    ls.set(0,2);
    personCopy.name = "李四";
    
    System.out.printf("引用传递_引用类型方法中=> " +
            "\n arr[0]=%d, chs[0]=%c,ls.get(0)=%d, person.name=%s \n", arr[0], chs[0], ls.get(0), personCopy.name);
}
public static class Person{
    public String name;
}

在这里插入图片描述

引用传递_引用类型方法执行前=> 
 arr[0]=1, chs[0]=1,ls.get(0)=1, person.name=张三 
引用传递_引用类型方法中=> 
 arr[0]=2, chs[0]=2,ls.get(0)=2, person.name=李四 
引用传递_引用类型方法结束后=> 
 arr[0]=2, chs[0]=2,ls.get(0)=2, person.name=李四 

通过底层原理理解

堆栈角度解释,Java是值传递。在Java中,当我们将一个变量从调用者的作用域传递到被调用的作用域时,传递的是变量的值(即其值,基本类型叫值,对象叫引用),而不是变量的内存地址。

值传递(基本数据类型传递)图解

对于基本数据类型,Java会进行栈上分配内存,传递的只是一份克隆数据,所以互不影响。

在这里插入图片描述

引用类型传递图解

对于引用数据类型,大部分情况下Java会进行堆上分配内存,栈上的person只是一个引用。person和personCopy是两个引用,但是都指向了同一个地址Person@60215eee,所以对这两个引用的对象内部进行修改导致的影响是一样的。

在这里插入图片描述

检验下你是否真的理解?

public static void main(String[] args) {
    Person person = new Person();
    person.name = "张三";
    System.out.printf("引用传递_引用类型方法执行前=> person.name=%s \n", person.name);
    引用传递_引用类型(person);
    System.out.printf("引用传递_引用类型方法结束后=> person.name=%s \n", person.name);
}
public static void 引用传递_引用类型(Person person){
    person = new Person();
    person.name = "李四";
}
public static class Person{
    public String name;
}

上面的代码【引用传递_引用类型方法结束后=>person.name=】输出什么?

  • 20
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值