java參数传递方式问题

java的參数传递方式到底是值传递还是引用传递,这一直是一个争论不休的问题,一直以来没有形成统一意见。

在这里,我也仅仅是说一说个人见解,不保证是对的,全当是抛砖引玉。

首先我的观点是java採用的是值传递,即使是引用作为实參也仅仅是将引用的值传递给另外一个引用,说白了还是值传递。

1、首先对于java基本类型(如int,double,byte)和不可变类型如(String)做參数一定是值传递,如以下程序:

public class Config {
public static void main(String[] args) {
int a1=9;
String s1="abc";
System.out.println("改变之前,a1="+a1+",s1="+s1);
Change(a1);
Change(s1);
System.out.println("改变之后,a1="+a1+",s1="+s1);
}
public static void Change(int a2) {
a2=12;
}
public static void Change(String s2) {
s2="def";
}
}

java參数传递方式问题

因为參数是值传递,编译器仅仅是将a1、s1的值传递给了a2、s2,当a2、s2的值发生变化时不会影响实參a1、s1的值,所以不变。

2、对于java对象的引用做參数,依旧是值传递

仅仅是此时的值传递,是将一个引用的值传递给还有一个引用,如以下所看到的:

public class Config {
public static void main(String[] args) {
StringBuffer s1=new StringBuffer("abc");
System.out.println("改变之前,s1="+s1);
Change(s1);
System.out.println("改变之后,s1="+s1);
}
public static void Change(StringBuffer s2) {
s2.append("def");
}
}

java參数传递方式问题

这时候大家可能就非常疑惑了,为什么在这里s1的值被改变了。

首先我们先明白一个概念,s1是一个引用而不是一个对象。new StringBuffer("abc")是一个对象,它被存放在内存中的堆里面。而s1是对new StringBuffer("abc")的引用,相当于C语言中的指针一样,里面存放着new StringBuffer("abc")的地址,s1被放在内存中的栈里面。s1指向new
StringBuffer("abc"),例如以下:

java參数传递方式问题

当s1作为实參时,是将s1的值复制给s2,s2也是一个引用,此时相当于将的s1中存放的地址复制给了s2,此时s2也指向了new StringBuffer("abc"),例如以下:

java參数传递方式问题

此时因为s2也存放了StringBuffer("abc")的地址,s2.append("def")相当于直接在StringBuffer("abc")后面插入了“def”,因此输出的s1结果发生了变化。

再看以下这个样例:

public class Config {
public static void main(String[] args) {
StringBuffer s1=new StringBuffer("abc");
System.out.println("改变之前,s1="+s1);
Change(s1);
System.out.println("改变之后,s1="+s1);
}
public static void Change(StringBuffer s2) {
s2=new StringBuffer("def");
}
}

java參数传递方式问题

可能大家看到这个结果会更加的不解,这个时候s1为啥没有发生变化呢??

首先,引用s1的值传递给了引用s2,此时s2存放着StringBuffer("abc")的地址,然后s2=new StringBuffer("def")这一步又将StringBuffer("def")的地址赋给了s2,也就是说s2中的地址发生了变化,不再指向StringBuffer("abc"),即s1、s2指向了不同的对象,所以s2之后的操作不再影响到s1。例如以下:

java參数传递方式问题

以上纯属个人见解,欢迎各种喷!!

上一篇:c++Volatile关键词


下一篇:python_way ,day5 模块,模块3 ,双层装饰器,字符串格式化,生成器,递归,模块倒入,第三方模块倒入,序列化反序列化,日志处理