學習Java的同學注意了!!!
學習進程中遇到甚么問題或想獲得學習資源的話,歡迎加入Java學習交換群,群號碼:183993990 我們1起學Java!
1. 棧(stack)與堆(heap)都是Java用來在Ram中寄存數據的地方。與C++不同,Java自動管理棧和堆,程序員不能直接地設置棧或堆。
2. 棧的優勢是,存取速度比堆要快,僅次于直接位于CPU中的寄存器。但缺點是,存在棧中的數據大小與生存期必須是肯定的,缺少靈活性。另外,棧數據可以同享,詳見第3點。堆的優勢是可以動態地分配內存大小,生存期也沒必要事前告知編譯器,Java的垃圾搜集器會自動收走這些不再使用的數據。但缺點是,由于要在運行時動態分配內存,存取速度較慢。
1 ==是判斷兩個對象是不是是同1個對象 2 equals是進行值的判斷 3 String a = new String( "aaa "); 4 String b = new String( "a "); 5 b += "aa "; 6 則 a==b //毛病 7 a.equals(b)//正確 8 9 10 11 12 13 14 除String和封裝器,equals()和“==”沒甚么區分 15 但String和封裝器重寫了equals(),所以在這里面,equals()指比較字符串或封裝對象對應的原始值是不是相等, "== "是比較兩個對象是不是為同1個對象
首先,我們先來看1下java中變量的語義:
java的變量有兩種語義,原始類型的變量是值語義(value),也就是說,你給1個原始類型變量賦值,就改變了這個數據值本身。對象類型的變量是援用語義,也就是說,給1個對象類型的變量賦值只是讓它指向另外一個對象,但不改變原來援用的那個對象的值。
然后,我們了解1下String的特性和java對Sting特別的處理方式:
《String的特性》
1、String類是final的,不可被繼承。
2、String類是的本質是字符數組char[], 并且其值不可改變。
3、String類對象有個特殊的創建的方式,就是直接指定比如String x = "abc","abc"就表示1個字符串對象。而x是"abc"對象的地址,也叫做"abc"對象的援用。
4、String對象可以通過“+”串連。串連后會生成新的字符串。
5、Java運行時會保護1個String Pool(String池),JavaDoc翻譯很模糊“字符串緩沖區”。String池用來寄存運行時中產生的各種字符串,并且池中的字符串的內容不重復。而1般對象不存在這個緩沖池,并且創建的對象僅僅存在于方法的堆棧區。
6、創建字符串的方式很多,歸納起來有3類:
其1,使用new關鍵字創建字符串,比如String s1 = new String("abc");
其2,直接指定。比如String s2 = "abc";
其3,使用串連生成新的字符串。比如String s3 = "ab" + "c";
《String對象的創建》
String對象的創建也有很多門道,關鍵是要明白其原理。
原理1:當使用任何方式來創建1個字符串對象s=X時,Java運行時(運行中JVM)會拿著這個X在String池中找是不是存在內容相同的字符串對象,如果不存在,則在池中創建1個字符串s,否則,不在池中添加。
原理2:Java中,只要使用new關鍵字來創建對象,則1定會(在堆區或棧區)創建1個新的對象。
原理3:使用直接指定或使用純字符串串連來創建String對象,則僅僅會檢查保護String池中的字符串,池中沒有就在池中創建1個,有則罷了!但絕不會在堆棧區再去創建該String對象。
原理4:使用包括變量的表達式來創建String對象,則不但會檢查保護String池,而且還會在堆棧區創建1個String對象。
《不可變類》
JAVA為了提高效力,對String類型進行了特別的處理---為string類型提供了串池
定義1個string類型的變量有兩種方式:
string name= "tom ";(String name="t"+"o"+"m"的效果和此處是相同的)
string name =new string( "tom ")
如果你使用了第1種方式,那末當你在聲明1個內容也是 "tom "的string時,它將使用串池里原來的那個內存,而不會重新分配內存,也就是說,string saname= "tom ",將會指向同1塊內存。而如果用第2種方式,不管串池里有無"tom",它都會在堆中重新分配1塊內存,定義1個新的對象。
另外關于string類型是不可改變的問題: string類型是不可改變的,也就是說,當你想改變1個string對象的時候,比如name= "madding " 那末虛擬機不會改變原來的對象,而是生成1個新的string對象,然后讓name去指向它,如果原來的那個 "tom "沒有任何對象去援用它,虛擬機的垃圾回收機制將接收它。
最后,關于定義String的堆棧問題
String s =new String()分析堆與棧,是先定義S,還是先new string()
1. String str1 = "abc";
System.out.println(str1 == "abc");
步驟:
1) 棧中開辟1塊空間寄存援用str1;
2) String池中開辟1塊空間,寄存String常量"abc";
3) 援用str1指向池中String常量"abc";
4) str1所指代的地址即常量"abc"所在地址,輸出為true;
2. String str2 = new String("abc");
System.out.println(str2 == "abc");
步驟:
1) 棧中開辟1塊空間寄存援用str2;
2) 堆中開辟1塊空間寄存1個新建的String對象"abc";
3) 援用str2指向堆中的新建的String對象"abc";
4) str2所指代的對象地址為堆中地址,而常量"abc"地址在池中,輸出為false;
3. String str3 = new String("abc");
System.out.println(str3 == str2);
步驟:
1) 棧中開辟1塊空間寄存援用str3;
2) 堆中開辟1塊新空間寄存另外1個(不同于str2所指)新建的String對象;
3) 援用str3指向另外新建的那個String對象 ;
4) str3和str2指向堆中不同的String對象,地址也不相同,輸出為false;
4. String str4 = "a" + "b";
System.out.println(str4 == "ab");
步驟:
1) 棧中開辟1塊空間寄存援用str4;
2) 根據編譯器合并已知量的優化功能,池中開辟1塊空間,寄存合并后的String常量"ab";
3) 援用str4指向池中常量"ab";
4) str4所指即池中常量"ab",輸出為true;
5. final String s = "a"; //注意:這里s用final修飾,相當于1個常量
String str5 = s + "b";
System.out.println(str5 == "ab");
步驟:
同4
6. String s1 = "a";
String s2 = "b";
String str6 = s1 + s2;
System.out.println(str6 == "ab");
步驟:
1) 棧中開辟1塊中間寄存援用s1,s1指向池中String常量"a",
2) 棧中開辟1塊中間寄存援用s2,s2指向池中String常量"b",
3) 棧中開辟1塊中間寄存援用str5,
4) s1 + s2通過StringBuilder的最后1步toString()方法還原1個新的String對象"ab",因此堆中開辟1塊空間寄存此對象,
5) 援用str6指向堆中(s1 + s2)所還原的新String對象,
6) str6指向的對象在堆中,而常量"ab"在池中,輸出為false
7. String str7 = "abc".substring(0, 2);
步驟:
1) 棧中開辟1塊空間寄存援用str7,
2) substring()方法還原1個新的String對象"ab"(不同于str6所指),堆中開辟1塊空間寄存此對象,
3) 援用str7指向堆中的新String對象,
8. String str8 = "abc".toUpperCase();
步驟:
1) 棧中開辟1塊空間寄存援用str6,
2) toUpperCase()方法還原1個新的String對象"ABC",池中并未開辟新的空間寄存String常量"ABC",
3) 援用str8指向堆中的新String對象
9.String s="abc";
String s1=s;
System.out.println(s1=="abc");
s=s+"hello";
System.out.println(s1=="abc");
System.out.println(s=="abc");
步驟:
1)棧中開辟1塊空間寄存s;
2)Sting池中開辟1塊空間用于寄存"abc",棧中開辟1塊空間寄存變量s1;
3)系統輸出true,在堆中開辟1塊空間用于寄存"abchello";
4)援用s指向堆中的"abchello";
5)系統輸出true,然后輸出false;
學習Java的同學注意了!!!
學習進程中遇到甚么問題或想獲得學習資源的話,歡迎加入Java學習交換群,群號碼:183993990 我們1起學Java!