start()方法纔是正確的線程啓動方式,run()方法僅僅是普通的方法調用。(常見面試題)
Thread和Runnable的關係:Thread 類本身實現了 Runnable 接口,並且持有 run 方法,但 Thread 類的 run 方法主體是空的,Thread 類的 run 方法通常是由子類的 run 方法重寫。
實現Runnable接口比繼承Thread類所具有的優勢:
(1)適合多個相同的程序代碼的線程去處理同一個資源。
(2)可以避免Java中的單繼承的限制。
(3)增加程序的健壯性,代碼可以被多個線程共享,代碼和數據獨立。
(4)線程池只能放入實現Runable或Callable類線程,不能直接放入繼承Thread的類。
線程安全概念:當多個線程訪問某一個類(對象或方法)時,這個類始終能表現出正確的行爲,那麼這個類(對象或方法)就是線程安全的。
線程安全的代碼會通過同步機制(比如加鎖)保證各個線程都可以正常且正確的執行,不會出現數據污染等意外情況。
加鎖機制常見的如:Synchronized
一個對象有一把鎖,如果要讓所有對象共用一個鎖,給方法加上static即可。因爲用 static 修改的方法或者變量,在該類的所有對象是具有相同的引用的。
可重入鎖:即該鎖可以被對象自己重新獲取自己的內部鎖,避免出現死鎖。
synchronized ()中的括號要傳入同一個對象纔有意義,可以傳入的對象例如有;this(多個線程中this不一定是同一個對象)、字節碼對象(一定是同一個對象,內存中只有一個字節碼對象,例如傳入String.class)、局部對象等。
實現多線程的方式有幾種?
一般來說是兩種,一種是繼承Thread類,另外一種是實現Runable接口。說得全面點還有一種是實現Callable接口,並與Future、線程池結合使用。
wait()與sleep()區別
- 對於sleep()方法,該方法是屬於Thread類中的。而wait()方法,則是屬於Object類中的。
- sleep()方法導致了程序暫停執行指定的時間,讓出cpu給其他線程,但是他的監控狀態依然保持者,當指定的時間到了又會自動恢復運行狀態。
- 在調用sleep()方法的過程中,線程不會釋放對象鎖。
- 而當調用wait()方法的時候,線程會放棄對象鎖,進入等待此對象的等待鎖定池,只有針對此對象調用notify()方法後本線程才進入對象鎖定池準備獲取對象鎖進入運行狀態。
volatile關鍵字可以保證可見性和有序性,不能保證原子性。
歡迎評論點贊收藏!您的支持是我創作的動力!感謝支持!