在java中,final关键字可以对类、方法以及变量进行修饰,这里对final修饰变量进行一些探讨。
final修饰的变量不可更改,其不可更改指的是其引用不可修改。看下列代码:
StringBuffer sb1 = new StringBuffer("sb");
StringBuffer sb2 = sb1;
final StringBuffer finalSb = sb2;
System.out.println(finalSb);//输出: sb
sb1.append(" sb1 append");
System.out.println(finalSb);//输出: sb sb1 append
sb2.append(" sb2 append");
System.out.println(finalSb);//输出: sb sb1 append sb2 append
在代码中,无论是对sb1还是sb2进行操作,均可以影响到finalSb的字面值,看似是finalSb → sb2 → sb1,这样一个变量指向另外一个变量,将变量的值复制传递下去。
这里进行一个猜想,原来是finalSb → sb2 → sb1,现在将sb2 → new StringBuffer(" new sb2"),finalSb的字面值是否变化呢?我们看代码输出:
StringBuffer sb1 = new StringBuffer("sb");
StringBuffer sb2 = sb1;
final StringBuffer finalSb = sb2;
System.out.println(finalSb);//输出: sb
sb1.append(" sb1 append");
System.out.println(finalSb);//输出: sb sb1 append
sb2.append(" sb2 append");
System.out.println(finalSb);//输出: sb sb1 append sb2 append
sb2 = new StringBuffer("new sb2");
System.out.println(finalSb);//输出: sb sb1 append sb2 append
sb2.append(" xx app");//对sb2进行操作
System.out.println(finalSb);//输出: sb sb1 append sb2 append
通过代码的输出可以看到,哪怕是将sb2重新指向了另外一个新的引用,finalSb的字面值并不会发生改变,对sb2进行的操作,也自然不会影响到finalSb的字面值。
sb1和原来的sb2操作时,可以影响到finalSb的值,新的sb2又不可以影响到finalSb的值,这是为什么呢?这里我画出示意图如下:
sb2并不是直接指向sb1,而是指向的是sb1在堆空间中new出来的对象,同样的,finalSb也不是指向的sb2,而是指向的sb2所指向的堆空间的对象,因为三者均是指向的同一个对象,所以,无论是对哪一个变量进行操作,均会影响到其字面值。
当执行 sb2 = new StringBuffer("new sb2") 时,sb2不再执行堆空间原来的对象,而是指向了新的对象,如图:
sb2已经指向了新的地址,故而对sb2进行怎么样的操作,finalSb的字面值均不会发生变化。
从以上两张图和代码可以看出,final所修饰的变量,其引用不可变(或者是其在内存中指向的地址不可更改),还有就是,基于引用本身的操作(如List类型的add,StringBuffer的append)不受影响。
另外,final修饰的变量还被赋予了编译常量的功能,如代码所示:
final String finalStrA = "str";
String strA = "str";
String strB = "str2";
String strC = finalStrA+"2";
String strD = strA+"2";
String strE = "str"+"2";//等同于 finalStrA+"2"
System.out.println(strB==strC);//输出: true
System.out.println(strB==strD);//输出: false
System.out.println(strB==strE);//输出: true
————————————————
版权声明:本文为CSDN博主「闲_风」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/a15123837995/article/details/83048772
网友评论