个人见解
栈需要知道被存储对象(不管是对象还是对象引用)的大小,而堆不需要。
因此对象引用存在栈中,而对象本身存储在堆中。
如同一本书,栈是目录这样的一个角色,堆是内容。
有了这样的思路,很容易推断出你想要的答案。
原子类型,数据就是自身。因此直接存在栈中。
一、栈内存 基础类型int, short, long, byte, float, double, boolean, char和对象引用 栈的共享特性 String str1 = "abc"; 1、编译器先处理String str1 = "abc";它会在栈中创建一个变量为str1的引用,然后查找栈中是否有abc这个值,如果没找到,就将abc存放进来,然后将str1指向abc。 2、 接着处理String str2 = "abc";在创建完b的引用变量后,因为在栈中已经有abc这个值,便将str2直接指向abc。这样,就出现了str1与str2同时均指向abc的情况。 二、堆内存 new、newarray、anewarray和multianewarray等指令建立 要注意: 我们在使用诸如String str = "abc";的格式定义类时,总是想当然地认为,创建了String类的对象str。担心陷阱!对象可能并没有被创建!而可能只是指向一个先前已经创建的 对象。只有通过new()方法才能保证每次都创建一个新的对象。 由于String类的immutable性质,当String变量需要经常变换其值时,应该考虑使用StringBuffer类,以提高程序效率。 三、 == 内存地址比对 String str1 = "abc"; String str3 = "abc"; System.out.println(str3 == str4); //flase str3值在栈内存中,str4值在堆内存中 String hello = "hello" ; String hel = "hel" ; String lo = "lo" ; System.out.println(hello == "hel" + "lo") ; //true //两个常量相加,先检测栈内存中是否有hello如有有,指向已有的栈中的hello空间 System.out.println(hello == "hel" + lo) ; //flase System.out.println(hello == hel + lo) ; //flase //lo是在常量池中,不检查栈内存,在堆中产生一个新的hello 四、 equals 值进行比对 public booleanequals(Object anObject) 将此字符串与指定的对象比较。当且仅当该参数不为 null,并且是与此对象表示相同字符序列的 String 对象时,结果才为 true。 String str5 = "abc"; System.out.println(str5.equals(str6)); //true str5的值str6的值比对 五、 intern 栈中值的内存地址 Public String intern() 当调用 intern 方法时 1、如果池已经包含一个等于此 String 对象的字符串(用equals(Object) 方法确定),则返回池中的字符串。 2、将此 String 对象添加到池中,并返回此 String 对象的引用。 String s7 = new String("abc") ; String s8 = "abc" ; System.out.println(s7 == s7.intern()) ;//flase; System.out.println(s8 == s7.intern() );//true 1.检查栈内存中有没有abc对象如果有 2.将s7指向pool中abc |