从深入理解Java虚拟机一书来说,是不存在这个说法的。从大部分百度而来的结果,就是说如下:
123,这个数字存在哪里? 这个从JVM来说,它是存在常量池中的。
按照缓存池的说法,int的缓存范围是-128~127,那我们用220来测试下结果。
private void cachePool(){ Integer a = Integer.valueOf(220); Integer b = Integer.valueOf(220); AppLog.i(String.valueOf(a == b)); Integer c = Integer.valueOf(123); Integer d = Integer.valueOf(123); AppLog.i(String.valueOf(c == d)); Integer e = 220; Integer f = 220; AppLog.i(String.valueOf(e == f)); Integer g = 123; Integer h = 123; AppLog.i(String.valueOf(g == h)); }
结果是false,true,false,true。
这里的理解是有误的。首先常量池存的是字符串,比如“abc”,abc存在常量池里面。s = "abc",这个时候,s并不在常量池。
在会过来说缓存池。在java中,当int的范围是【-128,127】时,JVM事先把这个范围内的数值,先存放在堆里面。所以对123的使用,缓存池的地址。
基本类型对应的缓冲池如下:
常量池只要针对字符串常量,这个上文有说到,字符串会在使用的时候,存放到常量池中。
String s1 = "abc"; String s2 = "abc"; AppLog.i(String.valueOf(s1 == s2));
结果是true,更进一步,把s2换成“abc”,还是true。也就是s1和s2都是常量池一个地址的引用。
最重要的是不可变性:
String 由final char[]value,由于final修饰,所以value不能指向其他数组,并且string内部没有修改value的方法,所以String是不可修改的。
hashcode可以缓存,应为不变。
StringPool机制,这个就是上面说的常量池。
String是线程安全的。
StringBuild不是线程安全
StringBuffer线程安全,内部有synchronized
String.intern(),这个方法会主动把字符串放入常量池。
Java 不能隐式执行向下转型,因为这会使得精度降低。
short s = 1; /*1是int类型,所以+以后转换为int类型,强转为short型*/ s = (short) (s+1); /*支持隐私转换*/ s +=1;
其他从JVM角度来讲,static的变量和类没有关系,类的实例是在java heap里面new出来的。而static 变量存在于方法区。类名只是为了找到这个全局变量的位置而已。(C++的说法,全局变量,功能类似)
静态方法也是独立于类的存在,所以静态方法必须实现。存在于JVM的方法区。
从JVM的角度看,只是一个命名方式而已,跟外部类,是完全不想干的2个类。