记录贴-java谈论关于是值传递还是引用传递引发的思考
1、问题
1、在讨论是值传递还是引用传递的时候突然有一个问题。在swag里面可不可以把mian里面的s2值修改掉,为什么。
public static void main(String[] args) {
String s1 = new String("one");
String s2 = new String("two");
swag(s1, s2);
System.out.println("s1 " + s1);
System.out.println("s2 " + s2);
}
private static void swag(String s1, String s2) {
s2 = s1;
s1 = "tree";
System.out.println("s1 " + s1);
System.out.println("s2 " + s2);
}
2、经过了思考,发现不行,无论怎么修改main中的s2还是two除了数组、封装类等一些方式,下面会说到。
2、思考
变量是什么由右边决定,右边决定左边,右边修改左边
1、上述程序的运行结果是
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-MwUjLL2G-1654863535902)(记录贴-java谈论关于是值传递还是引用传递引发的思考.assets/image-20220610194111835.png)]
说明s2在swag中并没有因为s1的赋值而改变,在main中s2的值还是two,这就是右边决定左边。
2、执行流程,假设main中的s1的地址是x01,s2的地址是x02,"tree"的地址是x03,那么在main中调用swag方法,那么现在swag方法中默认的s1是x01,s2是x02,但是这个时候 s2 = s1;那么现在s2的值就变成x02了,意思就是现在swag里面的S2已经和main里面的S2不一样了,因为swag里面的S2指向的是s1的地址,相当于说s2就是s1了,当执行到s1="tree"的时候,main里面的s1依旧不改变,因为现在swag里面的s1也不是s1了,现在swag的s1指向的是x03了,相当于说s1就是tree了,所以怎么修改都不会影响到s1。
3、因为现在swag的s2指向了x01也就是和main s1指向是一样了。那么现在令swag的s2 = “four”,那么main的s1会变成four吗?答案是不会,因为右边改变左边,当swag的s2 = "four"的时候,swag的s2不再是s2不再是main的s1了,假设four的地址是x04,那么现在的swag 的s2指向的是x04了,跟main的s1无关了。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-NJ0YZWzk-1654863535903)(记录贴-java谈论关于是值传递还是引用传递引发的思考.assets/image-20220610200022875.png)]
3、例2
public class Test1 {
public static void main(String[] args) {
Node a = new Node("one");
Node a2 = new Node("two");
swag(a ,a2);
System.out.println("");
System.out.println("a" + a.a);
System.out.println("a2" + a2);
}
private static void swag(Node a, Node a2) {
a2 = a;
a.a = "three";
System.out.println("a" + a.a);
System.out.println("a2" + a2);
}
}
class Node {
Node (String a) {
this.a = a;
}
public String a;
@Override
public String toString() {
return "Node{" +
"a='" + a + '\'' +
'}';
}
}
1、例2和上述情况特殊,特殊在swag的a.a = “three”,因为这步是可以改变main中的a的,但还是哪个原则右边决定这边,因为在这里面,a的地址并没有改变它指向的地址跟main中的a指向的地址是一样,但是a这个大地址里面还有许多小地址的指向,也就是a这个大地址里面还有一个String类型的指向,当a.a = "three"的时候,大地址里面的String地址发生了改变,但是大地址没有改变,swag还是指向main的a地址,所以在swag中a中还是指向的是main的a他们指向的是同一个地址。数组情况也是,数组的情况是,整体是一个地址,里面每一个空间又是其他的地址,所以数组这些大地址里面又有小地址的时候,在传给其他方法使用的时候,在方法中修改大地址里面的小地址,原来的也会一起发现改变。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-m9Ln97ak-1654863535903)(记录贴-java谈论关于是值传递还是引用传递引发的思考.assets/image-20220610201212710.png)]
4、扩展
1、因为是右边决定左边,右边改变左边。
2、所以就可能产生垃圾,因为
String a = "a";
String b = "b";
b = a;
在上述这三行代码中,b=a。右边决定左边。b已经变成a了,那么原来的"b"的地址也就没有人引用了,但是它又是占空间的。
String b = “b”;
b = a;
在上述这三行代码中,b=a。右边决定左边。b已经变成a了,那么原来的"b"的地址也就没有人引用了,但是它又是占空间的。
3、所以多次使用一个值或者变量的时候,要给他封装起来成为一个final static 或者变成枚举类,这样就可以减少垃圾的产生和空间的占用。