前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java到底是引用传递还是值传递?

Java到底是引用传递还是值传递?

作者头像
黑洞代码
发布2021-01-14 15:31:11
8010
发布2021-01-14 15:31:11
举报
文章被收录于专栏:落叶飞翔的蜗牛

Java 中只有值传递!

首先回顾一下在程序设计语言中有关将参数传递给方法(或函数)的一些专业术语。 按值调用(call by value)表示方法接收的是调用者提供的值,而按引用调用(call by reference)表示方法接收的是调用者提供的变量地址。一个方法可以修改传递引用所对应的变量值,而不能修改传递值调用所对应的变量值。它用来描述各种程序设计语言(不只是 Java)中方法参数传递方式。

值传递:作为参数传递时只传递了值,参数本身没有传递,如果被传递的方法改变传递后的变量参数值,原变量不会改变。在JAVA中显式的值传递参数是各类型变量例如:int double char String Integer 等都是。

引用传递:作为参数传递时传递的是参数本身,当被传递方法改变传递后的变量参数值,原参数也会发生改变。一般来说我们自定义的类基本都属于这种情况,那么为什么String类型变量本质也是对象但是没有出现这种情况,这个后边会详细解释。

注意:在C语言里有引用传递这种说法,但是在JAVA里虽然有类似效果但是实质上不存在任何引用传递,我们看到的类似引用传递的效果本质上还是值传递,原因下边慢慢看。

堆 & 栈 的定义在之前已经说过,那么这里说一下,基本类型变量 int boolean char float double等这些基本类型在定义后其数据是存在栈里这里说过,那么基本类型变量传递的时候是只传递了一个值,新变量本身除了值和原型一致就没有任何其他关系,所以基本类型变量传递是值传递。(注意虽然java是面向对象编程,但是基本类型变量不是对象(数组是对象),为了解决这个问题有了封装类。)

首先,你需要了解下java变量的分类:java中的变量分为

基本类型

接口类型

类类型

数组类型

其中后面三种统称为引用类型,而基本类型分为三种,

数字类型

boolean

returnAddress

数字类型又分为

浮点类型

整数类型

浮点

Java 程序设计语言总是采用按值调用。也就是说,方法得到的是所有参数值的一个拷贝,也就是说,方法不能修改传递给它的任何参数变量的内容。

下面通过 3 个例子来给大家说明

代码语言:javascript
复制
public static void main(String[] args) {
    int num1 = 10;
    int num2 = 20;
    // 两数交换的方法
    swap(num1, num2);

    System.out.println("num1 = " + num1);
    System.out.println("num2 = " + num2);
}

public static void swap(int a, int b) {
    int temp = a;
    a = b;
    b = temp;

    System.out.println("a = " + a);
    System.out.println("b = " + b);
}

执行结果如下:

代码语言:javascript
复制
a = 20
b = 10
num1 = 10
num2 = 20

原理解析:

在swap方法中,a、b 的值进行交换,并不会影响到 num1、num2。因为,a、b 中的值,只是从 num1、num2 的复制过来的。也就是说,a、b 相当于 num1、num2 的副本,副本的内容无论怎么修改,都不会影响到原件本身。

通过上面例子,我们已经知道了一个方法不能修改一个基本数据类型的参数,而对象引用作为参数就不一样。

代码语言:javascript
复制
public static void main(String[] args) {
    int[] arr = { 1, 2, 3, 4, 5 };
    System.out.println(arr[0]);
    change(arr);
    System.out.println(arr[0]);
}

public static void change(int[] array) {
    // 将数组的第一个元素变为0
    array[0] = 0;
}

执行结果:

代码语言:javascript
复制
1
0

原理解析:

array 被初始化 arr 的拷贝也就是一个对象的引用,也就是说 array 和 arr 指向的是同一个数组对象。因此,外部对引用对象的改变会反映到所对应的对象上。

通过上面的例子我们已经看到,实现一个改变对象参数状态的方法并不是一件难事。理由很简单,方法得到的是对象引用的拷贝,对象引用及其他的拷贝同时引用同一个对象。

很多程序设计语言(特别是,C++和 Pascal)提供了两种参数传递的方式:值调用和引用调用。有些程序员(甚至本书的作者)认为 Java 程序设计语言对对象采用的是引用调用,实际上,这种理解是不对的。由于这种误解具有一定的普遍性,所以下面给出一个反例来详细地阐述一下这个问题。

代码语言:javascript
复制
public class Test {
  public static void main(String[] args) {
    // TODO Auto-generated method stub
    Student s1 = new Student("小张");
    Student s2 = new Student("小李");
    Test.swap(s1, s2);
    System.out.println("s1:" + s1.getName());
    System.out.println("s2:" + s2.getName());
  }

  public static void swap(Student x, Student y) {
    Student temp = x;
    x = y;
    y = temp;
    System.out.println("x:" + x.getName());
    System.out.println("y:" + y.getName());
  }
}

执行结果:

代码语言:javascript
复制
x:小李
y:小张
s1:小张
s2:小李

原理解析:

交换之前:

交换之后:

通过上面两张图可以很清晰的看出:方法并没有改变存储在变量 s1 和 s2 中的对象引用。swap 方法的参数 x 和 y 被初始化为两个对象引用的拷贝,这个方法交换的是这两个拷贝。

Java 程序设计语言对对象采用的不是引用调用,实际上,对象引用是按值传递的。

下面再总结一下 Java 中方法参数的使用情况:

  • 一个方法不能修改一个基本数据类型的参数(即数值型或布尔型)。
  • 一个方法可以改变一个对象参数的状态。
  • 一个方法不能让对象参数引用一个新的对象。
本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2020-12-19,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 落叶飞翔的蜗牛 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档