OBJECT
clone();
一、Java中有兩種方式創(chuàng)建對象:1、使用new操作符創(chuàng)建對象;2、使用clone方法復(fù)制對象,并且clone要比創(chuàng)建一個新的對象的效率要高。
二、clone分為淺拷貝和深拷貝。淺拷貝指僅僅拷貝對象本身,如果對象中包含引用的對象,則不會被拷貝,深拷貝則會將引用對象一起拷貝。
三、要實現(xiàn)clone需要實現(xiàn)cloneable接口。如果沒有實現(xiàn)接口,直接clone則會拋出CloneNotSupportedException異常。
hashcode();
如何判別在集合中是否已經(jīng)存在該對象了?一般我們會用到equals()方法,,如果采用equals方法去逐一比較,效率必然是一個問題。此時,我們就需要用到hasncode()方法。實際上在HashMap的具體實現(xiàn)中會用一個table保存已經(jīng)存進去的對象的hashcode值,如果table中沒有該hashcode值,它就可以直接存進去,不用再進行任何比較了;如果存在該hashcode值, 就調(diào)用它的equals方法與新元素進行比較,相同的話就不存了,不相同就散列其它的地址,所以這里存在一個沖突解決的問題,這樣一來實際調(diào)用equals方法的次數(shù)就大大降低了,說通俗一點:Java中的hashCode方法就是根據(jù)一定的規(guī)則將與對象相關(guān)的信息(比如對象的存儲地址,對象的字段等)映射成一個數(shù)值,這個數(shù)值稱作為散列值。
不同的對象可能會產(chǎn)生相同的hashcode值,如果兩個對象的hashcode值不等,則必定是兩個不同的對象,所以,一般來說可以用hashcode()判斷兩個對象不相等,如果要判斷兩個對象相等需要使用到equal()方法。
notify()
notify方法只喚醒一個等待(對象的)線程并使該線程開始執(zhí)行。所以如果有多個線程等待一個對象,這個方法只會喚醒其中一個線程,選擇哪個線程取決于操作系統(tǒng)對多線程管理的實現(xiàn)。notifyAll 會喚醒所有等待(對象的)線程,盡管哪一個線程將會第一個處理取決于操作系統(tǒng)的實現(xiàn)。如果當前情況下有多個線程需要被喚醒,推薦使用notifyAll 方法。比如在生產(chǎn)者-消費者里面的使用,每次都需要喚醒所有的消費者或是生產(chǎn)者,以判斷程序是否可以繼續(xù)往下執(zhí)行
String
string.intern();
? ? ? jdk7 版本對 intern 操作和常量池都做了一定的修改。主要包括2點:
1、將String常量池從Perm區(qū)移動到了Java Heap區(qū)
2、String#intern方法時,如果存在堆中的對象,會直接保存對象的引用,而不會重新創(chuàng)建對象。
在 Jdk6 以及以前的版本中,字符串的常量池是放在堆的Perm區(qū)的,Perm區(qū)是一個類靜態(tài)的區(qū)域,主要存儲一些加載類的信息,常量池,方法片段等內(nèi)容,默認大小只有4m,一旦常量池中大量使用 intern 是會直接產(chǎn)生java.lang.OutOfMemoryError:PermGen space錯誤的
以下的程序作為理解:
String s =new String("1");
s.intern();
String s2 ="1";
System.out.println(s == s2);
String s3 =new String("1") +new String("1");
s3.intern();
String s4 ="11";
System.out.println(s3 == s4);
在jdk1.6以前結(jié)果為 false,false
在 jdk7 的版本中,false,true
AbstractStringBuilder
最大容量為Integer.MAX_VALUE-8
StringBuffer和StringBuilder
兩個默認的初始容量均為為16,前者是線程安全的因為它的方法均添加了鎖,
Thread
最小的權(quán)重是1,默認的權(quán)重是5,最大的權(quán)重為10
在thread類中,sleep,yield,join方法
(1)最簡單的區(qū)別是,wait方法依賴于同步,而sleep方法可以直接調(diào)用。而更深層次的區(qū)別在于sleep方法只是暫時讓出CPU的執(zhí)行權(quán),并不釋放鎖。而wait方法則需要釋放鎖。
(2)yield方法的作用是暫停當前線程,以便其他線程有機會執(zhí)行,不過不能指定暫停的時間,并且也不能保證當前線程馬上停止。yield方法只是將Running狀態(tài)轉(zhuǎn)變?yōu)镽unnable狀態(tài)。
(3)join方法的作用是父線程等待子線程執(zhí)行完成后再執(zhí)行,換句話說就是將異步執(zhí)行的線程合并為同步的線程
線程
wait方法是一個本地方法,其底層是通過一個叫做監(jiān)視器鎖的對象來完成的。
如何獲取monitor對象所有權(quán)?Java中只能通過Synchronized關(guān)鍵字來完成
1)調(diào)用wait方法后,線程是會釋放對monitor對象的所有權(quán)的。
2)一個通過wait方法阻塞的線程,必須同時滿足以下兩個條件才能被真正執(zhí)行:
線程需要被喚醒(超時喚醒或調(diào)用notify/notifyll)。
線程喚醒后需要競爭到鎖(monitor)