java-final关键字修饰变量

       在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的值,这是为什么呢?这里我画出示意图如下:

        java-final关键字修饰变量_第1张图片

       sb2并不是直接指向sb1,而是指向的是sb1在堆空间中new出来的对象,同样的,finalSb也不是指向的sb2,而是指向的sb2所指向的堆空间的对象,因为三者均是指向的同一个对象,所以,无论是对哪一个变量进行操作,均会影响到其字面值。

       当执行 sb2 = new StringBuffer("new sb2") 时,sb2不再执行堆空间原来的对象,而是指向了新的对象,如图:

     java-final关键字修饰变量_第2张图片

       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

 

 

你可能感兴趣的:(java基础,java,final,变量)