第二種方式:直接繼承Thread類創(chuàng)建對(duì)象
1.Thread子類無法再從其它類繼承(java語言單繼承)。
2.編寫簡單,run()方法的當(dāng)前對(duì)象就是線程對(duì)象,可直接操作。
在實(shí)際應(yīng)用中,幾乎都采取第一種方式
我們用代碼來模擬鐵路售票系統(tǒng),實(shí)現(xiàn)通過四個(gè)售票點(diǎn)發(fā)售某日某次列車的100張車票,一個(gè)售票點(diǎn)用一個(gè)線程表示。
我們首先這樣編寫這個(gè)程序:
Java代碼 class ThreadTest extends Thread{
private int ticket = 100;
public void run(){
while(true){
if(ticket 》 0){
System.out.println(Thread.currentThread().getName() +
“is saling ticket” + ticket--);
}else{
break;
}
}
}
}
main測試類:
Java代碼 public class ThreadDome1{
public static void main(String[] args){
ThreadTest t = new ThreadTest();
t.start();
t.start();
t.start();
t.start();
}
}
上面的代碼中,我們用ThreadTest類模擬售票處的售票過程,run方法中的每一次循環(huán)都將總票數(shù)減1,模擬賣出一張車票,同時(shí)該車票號(hào)打印出來,直接剩余的票數(shù)到零為止。在ThreadDemo1類的main方法中,我們創(chuàng)建了一個(gè)線程對(duì)象,并重復(fù)啟動(dòng)四次,希望通過這種方式產(chǎn)生四個(gè)線程。從運(yùn)行的結(jié)果來看我們發(fā)現(xiàn)其實(shí)只有一個(gè)線程在運(yùn)行,這個(gè)結(jié)果告訴我們:一個(gè)線程對(duì)象只能啟動(dòng)一個(gè)線程,無論你調(diào)用多少遍start()方法,結(jié)果只有一個(gè)線程。
我們接著修改ThreadDemo1,在main方法中創(chuàng)建四個(gè)Thread對(duì)象:
Java代碼 public class ThreadDemo1{
public static void main(String[] args){
new ThreadTest().start();
new ThreadTest().start();
new ThreadTest().start();
new ThreadTest().start();
}
}
Java代碼 class ThreadTest extends Thread{
private int ticket = 100;
public void run(){
while(true){
if(ticket 》 0){
System.out.println(Thread.currentThread().getName() +
“ is saling ticket” + ticket--);
}else{
break;
}
}
}
}
從結(jié)果上看每個(gè)票號(hào)都被打印了四次,即四個(gè)線程各自賣各自的100張票,而不去賣共同的100張票。這種情況是怎么造成的呢?我們需要的是,多個(gè)線程去處理同一個(gè)資源,一個(gè)資源只能對(duì)應(yīng)一個(gè)對(duì)象,在上面的程序中,我們創(chuàng)建了四個(gè)ThreadTest對(duì)象,就等于創(chuàng)建了四個(gè)資源,每個(gè)資源都有100張票,每個(gè)線程都在獨(dú)自處理各自的資源。
經(jīng)過這些實(shí)驗(yàn)和分析,可以總結(jié)出,要實(shí)現(xiàn)這個(gè)鐵路售票程序,我們只能創(chuàng)建一個(gè)資源對(duì)象,但要?jiǎng)?chuàng)建多個(gè)線程去處理同一個(gè)資源對(duì)象,并且每個(gè)線程上所運(yùn)行的是相同的程序代碼。在回顧一下使用接口編寫多線程的過程。
Java代碼 public class ThreadDemo1{
public static void main(String[] args){
ThreadTest t = new ThreadTest();
new Thread(t).start();
new Thread(t).start();
new Thread(t).start();
new Thread(t).start();
}
}
Java代碼 class ThreadTest implements Runnable{
private int tickets = 100;
public void run(){
while(true){
if(tickets 》 0){
System.out.println(Thread.currentThread().getName() +
“ is saling ticket ” + tickets--);
}
}
}
}
上面的程序中,創(chuàng)建了四個(gè)線程,每個(gè)線程調(diào)用的是同一個(gè)ThreadTest對(duì)象中的run()方法,訪問的是同一個(gè)對(duì)象中的變量(tickets)的實(shí)例,這個(gè)程序滿足了我們的需求。在Windows上可以啟動(dòng)多個(gè)記事本程序一樣,也就是多個(gè)進(jìn)程使用同一個(gè)記事本程序代碼。
可見,實(shí)現(xiàn)Runnable接口相對(duì)于繼承Thread類來說,有如下顯著的好處:
(1)適合多個(gè)相同程序代碼的線程去處理同一資源的情況,把虛擬CPU(線程)同程序的代碼,數(shù)據(jù)有效的分離,較好地體現(xiàn)了面向?qū)ο蟮脑O(shè)計(jì)思想。
(2)可以避免由于Java的單繼承特性帶來的局限。我們經(jīng)常碰到這樣一種情況,即當(dāng)我們要將已經(jīng)繼承了某一個(gè)類的子類放入多線程中,由于一個(gè)類不能同時(shí)有兩個(gè)父類,所以不能用繼承Thread類的方式,那么,這個(gè)類就只能采用實(shí)現(xiàn)Runnable接口的方式了。
?
(3)有利于程序的健壯性,代碼能夠被多個(gè)線程共享,代碼與數(shù)據(jù)是獨(dú)立的。當(dāng)多個(gè)線程的執(zhí)行代碼來自同一個(gè)類的實(shí)例時(shí),即稱它們共享相同的代碼。多個(gè)線程操作相同的數(shù)據(jù),與它們的代碼無關(guān)。當(dāng)共享訪問相同的對(duì)象是,即它們共享相同的數(shù)據(jù)。當(dāng)線程被構(gòu)造時(shí),需要的代碼和數(shù)據(jù)通過一個(gè)對(duì)象作為構(gòu)造函數(shù)實(shí)參傳遞進(jìn)去,這個(gè)對(duì)象就是一個(gè)實(shí)現(xiàn)了Runnable接口的類的實(shí)例。
四核四線程和四核八線程的區(qū)別
4核4線就是說CPU有4個(gè)物理核心,所以任務(wù)管理器里就顯示出4張CPU圖表。
八線程,可以有2種情況, 物理4核和物理8核。
8核8線和前面4核4線一樣,4核8線 就是說,使用了超線程技術(shù),把一個(gè)物理核心,模擬成 2個(gè)邏輯核心,所以任務(wù)管理器會(huì)顯示出8張CPU表。
超線程技術(shù)讓(P4)處理器增加5%的裸晶面積,就可以換來15%~30%的效能提升。但實(shí)際上,在某些程式或未對(duì)多線程編譯的程式而言,超線程反而會(huì)降低效能。除此之外,超線程技術(shù)亦要操作系統(tǒng)的配合,普通支持多處理器技術(shù)的系統(tǒng)亦未必能充分發(fā)揮該技術(shù)。例如Windows 2000,英特爾并不鼓勵(lì)使用者在此系統(tǒng)中利用超線程。原先不支持多核心的Windows XP Home Edition卻支持超線程技術(shù)。
評(píng)論