string 是对象,所以==对于string来说,是对比的内存地址;
解释第一个:a==c//为true。 因为b被final修饰,那么它本身是一个常量,如果你原意的话,你可去看看编译后的class文件。里面变量c = "hello2",底层原理是java进行了优化。会查找内存中是否存在"hello2",如果存在,那么就将地址赋予c,很明显把a的内存地址赋予了c。所以才会a==c得到结果true
解释第二个:a==e//为false。因为string的+2,底层原理是,StringBuffer.append("2").toString(); 很明显,地址发生了改变,所以a==e得到结果会为false
由于string类比较特殊,jdk专门做了缓存优化。
原来Java运行时会维护一个String Pool(String池)。String池用来存放运行时中产生的各种字符串,并且池中的字符串的内容不重复。而一般对象不存在这个缓冲池,并且创建的对象仅仅存在于方法的堆栈区。
也就是说需要看string创建的方式:
1 当使用任何方式来创建一个字符串对象s时,Java运行时(运行中JVM)会拿着这个X在String池中找是否存在内容相同的字符串对象,如果不存在,则在池中创建一个字符串s,否则,不在池中添加。
2 Java中,只要使用new关键字来创建对象,则一定会(在堆区或栈区)创建一个新的对象。
3 使用直接指定或者使用纯字符串串联来创建String对象,则仅仅会检查维护String池中的字符串,池中没有就在池中创建一个,有则罢了!但绝不会在堆栈区再去创建该String对象。
4 使用包含变量的表达式来创建String对象,则不仅会检查维护String池,而且还会在堆栈区创建一个String对象。
另外,String的intern()方法是一个本地方法,定义为public native String intern(); intern()方法的价值在于让开发者能将注意力集中到String池上。当调用 intern 方法时,如果池已经包含一个等于此 String 对象的字符串(该对象由 equals(Object) 方法确定),则返回池中的字符串。否则,将此 String 对象添加到池中,并且返回此 String 对象的引用。