Java:值传递or引用传递

前几天在一套面试题中遇到了一道关于Java的参数传递方式的问题,当时没有回答正确。后来在查阅资料后对这个问题有了详细的了解,在此记录一下。

在给类编程语言中,参数传递的方式有两种。一种是值传递,另一种是引用传递。首先来明确一下两种传递方式具体是什么意思。

值传递是指在方法调用时,传递给方法的参数,实际上是将原来的变量复制一份之后,将复制品传递给了被调用的方法。因为被调用方法得到的是原变量的一个拷贝,所以无论这个方法怎么修改参数,都不会改变原变量的值。

而引用传递是指在方法调用时,是将原变量的引用,也就是变量在内存中的地址空间,传给了被调用方法。因此,被调用方法中的参数和原变量指向的是同一段内存空间。当被调用方法修改参数所指的内存空间内部的值时,原变量的值也会改变。

对于Java的参数传递方式,有两种说法,有人说Java只有值传递,有人说Java既有值传递,又有引用传递。其实Java的传递方式到底叫什么并不重要,重要的是了解在不同的条件下,Java进行参数传递时,内存中到底发生了什么样的变化。

参数传递时内存的改变情况与所传递参数的类型有关。我们知道Java有两类变量类型:基本类型,比如int,float,double等;引用类型:比如数组,String,ArrayList等。这两类数据类型,在内存分配上的区别在于:基本类型的变量名所指向的内存中,存储的就是变量本身的值;而引用类型的变量名所指向的内存中,存储的是一个内存地址,这个内存地址内存储的才是引用类型本身。

对于基本类型,Java就是讲原类型复制一份,将拷贝传递给被调用函数。举例如下:

public class MyTest {

	public static void change(int b) {
		b = 6;
	}

	public static void main(String[] args) {
		int a = 5;
		change(a);
		System.out.println("a=" + a);// 输出a=5
	}

}
可见,我们在change()方法中对b进行了重新复制,但并没有影响a的值。两者在内存中的分配方式如下:



对于引用类类型,可以确定的是Java是将其在内存中的地址传递给了被调用方法。但是,当方法对参数进行改变时,是否会修改原参数则需要分情况讨论。

情况一:直接修改参数内部的值,这时,修改会改变原变量的值:

public class MyTest {

	public static void change(int[] b) {
		for(int i = 0; i<b.length; i++){
			b[i] = b[i] * 2;
		}
	}

	public static void main(String[] args) {
		int[] a = new int[]{1,2,3};
		change(a);
		for(int i = 0; i<a.length; i++){
			System.out.print(a[i] + " ");
		}
		// 输出结果为2 4 6
	}
}
内存中的情况为:


情况二:在被调用方法内部,对参数进行了重新赋值。在这种情况向,相当于在内存中新建了一个该引用类型,然后将参数指向了这个引用类型。因此被调用方法中的参数与原变量完全是两个东西,自然不会影响。
public class MyTest {

	public static void change(int[] b) {
		b = new int[]{2,4,6};
	}

	public static void main(String[] args) {
		int[] a = new int[]{1,2,3};
		change(a);
		for(int i = 0; i<a.length; i++){
			System.out.print(a[i] + " ");
		}
		// 输出结果为1 2 3
	}
}
内存中的情况为:

情况三:String类。String类也是一个引用类,但是它与别的引用类的区别在于,String类是一个不可变类,在定义时使用了final修饰。如果要对String类型进行改变,那么就会新建一个String类,所以也不会改变原变量的值。举例如下:

public class MyTest {

	public static void change(String b) {
		b += "cde";
	}

	public static void main(String[] args) {
		String s = "abc";
		change(s);
		System.out.println(s);
		// 输出结果为abc
	}
}


public class MyTest {

	public static void change(String b) {
		b = new String("cde");
	}

	public static void main(String[] args) {
		String s = "abc";
		change(s);
		System.out.println(s);
		// 输出结果为abc
	}
}



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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值