在任何一门编程语言中,都离不开方法或者说函数,其实函数就是方法。java也同样是如此。下面我就来介绍一下java中方法的参数传递问题。
首先在java中没有引用传递,只有值传递。但是值传递分两种:1.字面值传递。2.地址值传递(很多地方叫引用传递)。
在java编程思想中提到了一点,java中基本数据类型由于大量使用,并且每个变量占用的内存空间十分小,所以不需要通过new关键字分配内存,而直接将值存入到栈空间中。
java对于基本数据类型和引用数据类型的内存分配原则是不同的。
值传递
在java中,基本数据类型
的方法参数传递是 字面值传递
。为什么会这么说呢?在java语言中基本数据类型和对象引用的句柄都是存放在栈空间中的。
基本数据类型会将变量名和字面值保存在同一块栈空间中,你访问这个变量就相当于直接访问这个字面值。
举个例子:
public class Test{
public static void main(String[] args){
int a=6;
System.out.println("主方法没有调用change()方法之前:"+a);
change(a);
System.out.println("主方法调用change()方法之后:"+a);
}
public static void change(int num){
System.out.println("add()方法内部没有修改之前:"+num);
num=100;
System.out.println("add()方法内部修改之后:"+num);
}
}
输出结果:
主方法没有调用change()方法之前:6
add()方法内部没有修改之前:6
add()方法内部修改之后:100
主方法调用change()方法之后:6
从以上的输出结果来看,change()方法中形参num虽然修改了值,但是并没有影响到实参a的值。
为什么会这样呢?
我们来看下,与之相对应的栈调用图:
上图就解释了为什么形参修改了值,但是实参不受影响。
地址值传递
在java中,引用数据类型
的方法参数传递是 地址值传递
。上文也说了,对象的句柄存放在栈中保存的是对象的地址。按照java中只有值传递的说法来看,就是将对象的地址值复制一份给形参。
举个例子:
public class Test{
public static void main(String[] args){
String a="abc";
System.out.println("主方法没有调用change()方法之前:"+a);
change(a);
System.out.println("主方法调用change()方法之后:"+a);
}
public static void change(String str){
System.out.println("add()方法内部没有修改之前:"+str);
str="zxy";
System.out.println("add()方法内部修改之后:"+str);
}
}
输出结果:
主方法没有调用change()方法之前:abc
add()方法内部没有修改之前:abc
add()方法内部修改之后:zxy
主方法调用change()方法之后:abc
从以上的输出结果来看,有人也许会问了,这不是和基本数据类型的效果类似吗?这里怎么就是地址值传递了?
为什么会这样呢?
让我们来看下,内存中是操作的呢?
上面虽然参数传递是地址值传递,但是由于String类的特殊性,在chagne()方法中还是没有操作到实参的对象。这也是为什么这里是地址值传递但是却打印出值传递的效果了。
需要注意的是,这里因为操作的是不可变类String,所以造成值传递效果,但是依旧是地址值传递。如果这里换成可变的对象,输出结果就不是这个样子了。
比如说这里换成StringBuffer那输出的结果就应该是:
主方法没有调用change()方法之前:abc
add()方法内部没有修改之前:abc
add()方法内部修改之后:zxy
主方法调用change()方法之后:zxy
好了,以上就是对java中方法的参数传递的介绍。希望对你有所帮助。