以前总是遇到一些不可思议的事,但是那是没有时间和精力去解决,只是先记着,唉,记着记着,就有去解释一下的冲动了。
引例:
public class FinalKeyWordTest {
public static void main(String[] args) {
final String str1 = "我是"+ 123;
final String str2 = "我是" +String.valueOf(123);
System.out.println(str1 == "我是123");
System.out.println(str2 == "我是123");
}
}
//运行结果:
//true
//false
//乍一看,为何第一句输出真,而后一句输出假呢?刚开始我也极端纳闷,想不通(不过还没到要跳楼的程度),
//资料查啊查才有“一点”眉目
//总结:首先,我们要清楚“.equals()”方法和“==”的区别,第一句说明了str1和后面的字符串是同一个东西。str1和
//str2没有太大的区别,只是定义str2时显式将数值123转换成字符串,但由于该变量的值需要调用String类方法,因
//此编译器无法编译时确定str2的值,str2不会被当成“宏变量”处理。(但str1会被当成宏变量处理,在编译时就会确
//定值。
很多人遇到过这种情况,在匿名内部类中使用一个内部类外的局部变量时,如果该局部变量不加final关键字修饰,会编译不通过,这是为什么呢?如果你知道答案,以下内容请勿看,免得浪费你宝贵的时间。
public class InlineClassTest {
public static void main(String[] args) {
final String str = "Java";
new Thread(new Runnable() {
public void run() {
for (int i = 0; i < 100; i++) {
System.out.println(str + i);
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}).start();
// 执行到这里,main方法结束,但启动的线程还在运行
}
}
小释:如果str变量没有加final,那么当main方法执行结束后,str变量会被回收或被销毁,那么子线程用的str就不存在了,所以Java是不允许这些矛盾出现的,所以此时final关键字就起作用了,该关键字允许内部类产生隐式的“闭包”,闭包将使得局部变量脱离它所在的方法继续存在。
附:java缓存小实例:
public class BufferMemory {
public static void main(String[] args) {
String str1= "改变世界,下一个就是U";
String str2 = "改变世界,下一个就是U";
System.out.println(str1 == str2);
}
}
运行结果:
true
小释:Java会缓存曾经用过的字符串直接量(本人估计这样的机制是想提高运行速度和节省空间吧)。
总结:以上都是一些很小的细节,但是,见叶知秋,我们可以从一些小小的细节去窥探java运行机制,其实还是蛮有意思的。
计划:研究java集合的实现细节。