Java多線程看這一篇就足夠了(吐血超詳細總結)


進程與線程

進程是程序的一次動態執行過程,它需要經歷從代碼加載,代碼執行到執行完畢的一個完整的過程,這個過程也是進程本身從產生,發展到最終消亡的過程。多進程操作系統能同時達運行多個進程(程序),由於 CPU 具備分時機制,所以每個進程都能循環獲得自己的CPU 時間片。由於 CPU 執行速度非常快,使得所有程序好像是在同時運行一樣。

多線程是實現並發機制的一種有效手段。進程和線程一樣,都是實現並發的一個基本單位。線程是比進程更小的執行單位,線程是進程的基礎之上進行進一步的划分。所謂多線程是指一個進程在執行過程中可以產生多個更小的程序單元,這些更小的單元稱為線程,這些線程可以同時存在,同時運行,一個進程可能包含多個同時執行的線程。進程與線程的區別如圖所示: 

Java中線程實現的方式

在 Java 中實現多線程有兩種手段,一種是繼承 Thread 類,另一種就是實現 Runnable 接口。下面我們就分別來介紹這兩種方式的使用。

實現 Runnable 接口

 1 package ljz; 
 2 class MyThread implements Runnable{ // 實現Runnable接口,作為線程的實現類 
 3     private String name ;       // 表示線程的名稱 
 4     public MyThread(String name){ 
 5         this.name = name ;      // 通過構造方法配置name屬性 
 6     } 
 7     public void run(){  // 覆寫run()方法,作為線程 的操作主體 
 8         for(int i=0;i<10;i++){ 
 9             System.out.println(name + "運行,i = " + i) ; 
10         } 
11     } 
12 }; 
13 public class RunnableDemo01{ 
14     public static void main(String args[]){ 
15         MyThread mt1 = new MyThread("線程A ") ;    // 實例化對象 
16         MyThread mt2 = new MyThread("線程B ") ;    // 實例化對象 
17         Thread t1 = new Thread(mt1) ;       // 實例化Thread類對象 
18         Thread t2 = new Thread(mt2) ;       // 實例化Thread類對象 
19         t1.start() ;    // 啟動多線程 
20         t2.start() ;    // 啟動多線程 
21     } 
22 };

程序運行結果: 

繼承 Thread 類

 1 class MyThread extends Thread{  // 繼承Thread類,作為線程的實現類 
 2     private String name ;       // 表示線程的名稱 
 3     public MyThread(String name){ 
 4         this.name = name ;      // 通過構造方法配置name屬性 
 5     } 
 6     public void run(){  // 覆寫run()方法,作為線程 的操作主體 
 7         for(int i=0;i<10;i++){ 
 8             System.out.println(name + "運行,i = " + i) ; 
 9         } 
10     } 
11 }; 
12 public class ThreadDemo02{ 
13     public static void main(String args[]){ 
14         MyThread mt1 = new MyThread("線程A ") ;    // 實例化對象 
15         MyThread mt2 = new MyThread("線程B ") ;    // 實例化對象 
16         mt1.start() ;   // 調用線程主體 
17         mt2.start() ;   // 調用線程主體 
18     } 
19 };

程序運行結果: 

從程序可以看出,現在的兩個線程對象是交錯運行的,哪個線程對象搶到了 CPU 資源,哪個線程就可以運行,所以程序每次的運行結果肯定是不一樣的,在線程啟動雖然調用的是 start() 方法,但實際上調用的卻是 run() 方法定義的主體。

Thread 類和 Runnable 接口

通過 Thread 類和 Runable 接口都可以實現多線程,那么兩者有哪些聯系和區別呢?下面我們觀察 Thread 類的定義。

public class Thread extends Object implements Runnable

從 Thread 類的定義可以清楚的發現,Thread 類也是 Runnable 接口的子類,但在Thread類中並沒有完全實現 Runnable 接口中的 run() 方法,下面是 Thread 類的部分定義。

 1 Private Runnable target; 
 2 public Thread(Runnable target,String name){ 
 3     init(null,target,name,0); 
 4 } 
 5 private void init(ThreadGroup g,Runnable target,String name,long stackSize){ 
 6     ... 
 7     this.target=target; 
 8 } 
 9 public void run(){ 
10     if(target!=null){ 
11         target.run(); 
12     } 
13 }

從定義中可以發現,在 Thread 類中的 run() 方法調用的是 Runnable 接口中的 run() 方法,也就是說此方法是由 Runnable 子類完成的,所以如果要通過繼承 Thread 類實現多線程,則必須覆寫 run()。

實際上 Thread 類和 Runnable 接口之間在使用上也是有區別的,如果一個類繼承 Thread類,則不適合於多個線程共享資源,而實現了 Runnable 接口,就可以方便的實現資源的共享。

線程的狀態變化

要想實現多線程,必須在主線程中創建新的線程對象。任何線程一般具有5種狀態,即創建,就緒,運行,阻塞,終止。下面分別介紹一下這幾種狀態:

  • 創建狀態 

在程序中用構造方法創建了一個線程對象后,新的線程對象便處於新建狀態,此時它已經有了相應的內存空間和其他資源,但還處於不可運行狀態。新建一個線程對象可采用Thread 類的構造方法來實現,例如 “Thread thread=new Thread()”。

  • 就緒狀態 

新建線程對象后,調用該線程的 start() 方法就可以啟動線程。當線程啟動時,線程進入就緒狀態。此時,線程將進入線程隊列排隊,等待 CPU 服務,這表明它已經具備了運行條件。

  • 運行狀態 

當就緒狀態被調用並獲得處理器資源時,線程就進入了運行狀態。此時,自動調用該線程對象的 run() 方法。run() 方法定義該線程的操作和功能。

  • 阻塞狀態 

一個正在執行的線程在某些特殊情況下,如被人為掛起或需要執行耗時的輸入/輸出操作,會讓 CPU 暫時中止自己的執行,進入阻塞狀態。在可執行狀態下,如果調用sleep(),suspend(),wait() 等方法,線程都將進入阻塞狀態,發生阻塞時線程不能進入排隊隊列,只有當引起阻塞的原因被消除后,線程才可以轉入就緒狀態。

  • 死亡狀態 

線程調用 stop() 方法時或 run() 方法執行結束后,即處於死亡狀態。處於死亡狀態的線程不具有繼續運行的能力。

在此提出一個問題,Java 程序每次運行至少啟動幾個線程?

回答:至少啟動兩個線程,每當使用 Java 命令執行一個類時,實際上都會啟動一個 JVM,每一個JVM實際上就是在操作系統中啟動一個線程,Java 本身具備了垃圾的收集機制。所以在 Java 運行時至少會啟動兩個線程,一個是 main 線程,另外一個是垃圾收集線程。

取得和設置線程的名稱

 1 class MyThread implements Runnable{ //實現Runnable接口 
 2     public void run(){ 
 3        for(int i=0;i<3;i++){ 
 4            System.Out.Println(Thread.currentThread().getName()+"運行, i="+i);  //取得當前線程的名稱 
 5        } 
 6   } 
 7 }; 
 8 
 9 public class ThreadDemo{ 
10 public static void main(String args[]){ 
11     MyThread my=new MyThread();  //定義Runnable子類對象 
12     new Thread(my).start;    //系統自動設置線程名稱 
13     new Thread(my,"線程A").start();  //手工設置線程名稱 
14   } 
15 };   

程序運行結果: 


線程的操作方法

剛才在分析自定義模式工作原理的時候其實就已經提到了,如果想要更改Glide的默認配

線程的強制運行

在線程操作中,可以使用 join() 方法讓一個線程強制運行,線程強制運行期間,其他線程無法運行,必須等待此線程完成之后才可以繼續執行。

 1 class MyThread implements Runnable{ // 實現Runnable接口 
 2     public void run(){  // 覆寫run()方法 
 3         for(int i=0;i<50;i++){ 
 4             System.out.println(Thread.currentThread().getName() 
 5                     + "運行,i = " + i) ;  // 取得當前線程的名字 
 6         } 
 7     } 
 8 }; 
 9 public class ThreadJoinDemo{ 
10     public static void main(String args[]){ 
11         MyThread mt = new MyThread() ;  // 實例化Runnable子類對象 
12         Thread t = new Thread(mt,"線程");     // 實例化Thread對象 
13         t.start() ; // 啟動線程 
14         for(int i=0;i<50;i++){ 
15             if(i>10){ 
16                 try{ 
17                     t.join() ;  // 線程強制運行 
18                 }catch(InterruptedException e){
19                 } 
20             } 
21             System.out.println("Main線程運行 --> " + i) ; 
22         } 
23     } 
24 };

程序運行結果: 


線程的休眠

在程序中允許一個線程進行暫時的休眠,直接使用 Thread.sleep() 即可實現休眠。

 1 class MyThread implements Runnable{ // 實現Runnable接口 
 2     public void run(){  // 覆寫run()方法 
 3         for(int i=0;i<50;i++){ 
 4             try{ 
 5                 Thread.sleep(500) ; // 線程休眠 
 6             }catch(InterruptedException e){
 7             } 
 8             System.out.println(Thread.currentThread().getName() 
 9                     + "運行,i = " + i) ;  // 取得當前線程的名字 
10         } 
11     } 
12 }; 
13 public class ThreadSleepDemo{ 
14     public static void main(String args[]){ 
15         MyThread mt = new MyThread() ;  // 實例化Runnable子類對象 
16         Thread t = new Thread(mt,"線程");     // 實例化Thread對象 
17         t.start() ; // 啟動線程 
18     } 
19 };

程序執行結果: 


中斷線程

當一個線程運行時,另外一個線程可以直接通過interrupt()方法中斷其運行狀態。

 1 class MyThread implements Runnable{ // 實現Runnable接口 
 2     public void run(){  // 覆寫run()方法 
 3         System.out.println("1、進入run()方法") ; 
 4         try{ 
 5             Thread.sleep(10000) ;   // 線程休眠10秒 
 6             System.out.println("2、已經完成了休眠") ; 
 7         }catch(InterruptedException e){ 
 8             System.out.println("3、休眠被終止") ; 
 9             return ; // 返回調用處 
10         } 
11         System.out.println("4、run()方法正常結束") ; 
12     } 
13 }; 
14 public class ThreadInterruptDemo{ 
15     public static void main(String args[]){ 
16         MyThread mt = new MyThread() ;  // 實例化Runnable子類對象 
17         Thread t = new Thread(mt,"線程");     // 實例化Thread對象 
18         t.start() ; // 啟動線程 
19         try{ 
20             Thread.sleep(2000) ;    // 線程休眠2秒 
21         }catch(InterruptedException e){ 
22             System.out.println("3、休眠被終止") ; 
23         } 
24         t.interrupt() ; // 中斷線程執行 
25     } 
26 };

程序運行結果是:


后台線程

在 Java 程序中,只要前台有一個線程在運行,則整個 Java 進程都不會消失,所以此時可以設置一個后台線程,這樣即使 Java 線程結束了,此后台線程依然會繼續執行,要想實現這樣的操作,直接使用 setDaemon() 方法即可。

 1 class MyThread implements Runnable{ // 實現Runnable接口 
 2     public void run(){  // 覆寫run()方法 
 3         while(true){ 
 4             System.out.println(Thread.currentThread().getName() + "在運行。") ; 
 5         } 
 6     } 
 7 }; 
 8 public class ThreadDaemonDemo{ 
 9     public static void main(String args[]){ 
10         MyThread mt = new MyThread() ;  // 實例化Runnable子類對象 
11         Thread t = new Thread(mt,"線程");     // 實例化Thread對象 
12         t.setDaemon(true) ; // 此線程在后台運行 
13         t.start() ; // 啟動線程 
14     } 
15 };

在線程類 MyThread 中,盡管 run() 方法中是死循環的方式,但是程序依然可以執行完,因為方法中死循環的線程操作已經設置成后台運行。

線程的優先級

在 Java 的線程操作中,所有的線程在運行前都會保持在就緒狀態,那么此時,哪個線程的優先級高,哪個線程就有可能會先被執行。

 1 class MyThread implements Runnable{ // 實現Runnable接口 
 2     public void run(){  // 覆寫run()方法 
 3         for(int i=0;i<5;i++){ 
 4             try{ 
 5                 Thread.sleep(500) ; // 線程休眠 
 6             }catch(InterruptedException e){
 7             } 
 8             System.out.println(Thread.currentThread().getName() 
 9                     + "運行,i = " + i) ;  // 取得當前線程的名字 
10         } 
11     } 
12 }; 
13 public class ThreadPriorityDemo{ 
14     public static void main(String args[]){ 
15         Thread t1 = new Thread(new MyThread(),"線程A") ;  // 實例化線程對象 
16         Thread t2 = new Thread(new MyThread(),"線程B") ;  // 實例化線程對象 
17         Thread t3 = new Thread(new MyThread(),"線程C") ;  // 實例化線程對象 
18         t1.setPriority(Thread.MIN_PRIORITY) ;   // 優先級最低 
19         t2.setPriority(Thread.MAX_PRIORITY) ;   // 優先級最高 
20         t3.setPriority(Thread.NORM_PRIORITY) ;  // 優先級最中等 
21         t1.start() ;    // 啟動線程 
22         t2.start() ;    // 啟動線程 
23         t3.start() ;    // 啟動線程 
24     } 
25 };

程序運行結果: 


從程序的運行結果中可以觀察到,線程將根據其優先級的大小來決定哪個線程會先運行,但是需要注意並非優先級越高就一定會先執行,哪個線程先執行將由 CPU 的調度決定。

線程的禮讓

在線程操作中,也可以使用 yield() 方法將一個線程的操作暫時讓給其他線程執行

 1 class MyThread implements Runnable{ // 實現Runnable接口 
 2     public void run(){  // 覆寫run()方法 
 3         for(int i=0;i<5;i++){ 
 4             try{ 
 5                 Thread.sleep(500) ; 
 6             }catch(Exception e){
 7             } 
 8             System.out.println(Thread.currentThread().getName() 
 9                     + "運行,i = " + i) ;  // 取得當前線程的名字 
10             if(i==2){ 
11                 System.out.print("線程禮讓:") ; 
12                 Thread.currentThread().yield() ;    // 線程禮讓 
13             } 
14         } 
15     } 
16 }; 
17 public class ThreadYieldDemo{ 
18     public static void main(String args[]){ 
19         MyThread my = new MyThread() ;  // 實例化MyThread對象 
20         Thread t1 = new Thread(my,"線程A") ; 
21         Thread t2 = new Thread(my,"線程B") ; 
22         t1.start() ; 
23         t2.start() ; 
24     } 
25 };

程序執行結果: 


同步以及死鎖

一個多線程的程序如果是通過 Runnable 接口實現的,則意味着類中的屬性被多個線程共享,那么這樣就會造成一種問題,如果這多個線程要操作同一個資源時就有可能出現資源同步問題。

解決方法:

同步代碼塊

1 synchronized(同步對象){ 
2  需要同步的代碼 
3
 1 class MyThread implements Runnable{ 
 2     private int ticket = 5 ;    // 假設一共有5張票 
 3     public void run(){ 
 4         for(int i=0;i<100;i++){ 
 5             synchronized(this){ // 要對當前對象進行同步 
 6                 if(ticket>0){   // 還有票 
 7                     try{ 
 8                         Thread.sleep(300) ; // 加入延遲 
 9                     }catch(InterruptedException e){ 
10                         e.printStackTrace() ; 
11                     } 
12                     System.out.println("賣票:ticket = " + ticket-- ); 
13                 } 
14             } 
15         } 
16     } 
17 }; 
18 public class SyncDemo02{ 
19     public static void main(String args[]){ 
20         MyThread mt = new MyThread() ;  // 定義線程對象 
21         Thread t1 = new Thread(mt) ;    // 定義Thread對象 
22         Thread t2 = new Thread(mt) ;    // 定義Thread對象 
23         Thread t3 = new Thread(mt) ;    // 定義Thread對象 
24         t1.start() ; 
25         t2.start() ; 
26         t3.start() ; 
27     } 
28 };

程序執行結果: 


同步方法

除了可以將需要的代碼設置成同步代碼塊外,也可以使用 synchronized 關鍵字將一個方法聲明為同步方法。

1 synchronized 方法返回值 方法名稱(參數列表){ 
2 
3
 1 class MyThread implements Runnable{ 
 2     private int ticket = 5 ;    // 假設一共有5張票 
 3     public void run(){ 
 4         for(int i=0;i<100;i++){ 
 5             this.sale() ;   // 調用同步方法 
 6         } 
 7     } 
 8     public synchronized void sale(){    // 聲明同步方法 
 9         if(ticket>0){   // 還有票 
10             try{ 
11                 Thread.sleep(300) ; // 加入延遲 
12             }catch(InterruptedException e){ 
13                 e.printStackTrace() ; 
14             } 
15             System.out.println("賣票:ticket = " + ticket-- ); 
16         } 
17 
18     } 
19 }; 
20 public class SyncDemo03{ 
21     public static void main(String args[]){ 
22         MyThread mt = new MyThread() ;  // 定義線程對象 
23         Thread t1 = new Thread(mt) ;    // 定義Thread對象 
24         Thread t2 = new Thread(mt) ;    // 定義Thread對象 
25         Thread t3 = new Thread(mt) ;    // 定義Thread對象 
26         t1.start() ; 
27         t2.start() ; 
28         t3.start() ; 
29     } 
30 };

程序執行結果: 


從程序運行的結果可以發現,此代碼完成了與之前同步代碼同樣的功能。

死鎖

同步可以保證資源共享操作的正確性,但是過多同步也會產生問題。例如,現在張三想要李四的畫,李四想要張三的書,張三對李四說“把你的畫給我,我就給你書”,李四也對張三說“把你的書給我,我就給你畫”兩個人互相等對方先行動,就這么干等沒有結果,這實際上就是死鎖的概念。

所謂死鎖,就是兩個線程都在等待對方先完成,造成程序的停滯,一般程序的死鎖都是在程序運行時出現的。

下面以一個簡單范例說明這個概念

 1 class Zhangsan{ // 定義張三類 
 2     public void say(){ 
 3         System.out.println("張三對李四說:“你給我畫,我就把書給你。”") ; 
 4     } 
 5     public void get(){ 
 6         System.out.println("張三得到畫了。") ; 
 7     } 
 8 }; 
 9 class Lisi{ // 定義李四類 
10     public void say(){ 
11         System.out.println("李四對張三說:“你給我書,我就把畫給你”") ; 
12     } 
13     public void get(){ 
14         System.out.println("李四得到書了。") ; 
15     } 
16 }; 
17 public class ThreadDeadLock implements Runnable{ 
18     private static Zhangsan zs = new Zhangsan() ;       // 實例化static型對象 
19     private static Lisi ls = new Lisi() ;       // 實例化static型對象 
20     private boolean flag = false ;  // 聲明標志位,判斷那個先說話 
21     public void run(){  // 覆寫run()方法 
22         if(flag){ 
23             synchronized(zs){   // 同步張三 
24                 zs.say() ; 
25                 try{ 
26                     Thread.sleep(500) ; 
27                 }catch(InterruptedException e){ 
28                     e.printStackTrace() ; 
29                 } 
30                 synchronized(ls){ 
31                     zs.get() ; 
32                 } 
33             } 
34         }else{ 
35             synchronized(ls){ 
36                 ls.say() ; 
37                 try{ 
38                     Thread.sleep(500) ; 
39                 }catch(InterruptedException e){ 
40                     e.printStackTrace() ; 
41                 } 
42                 synchronized(zs){ 
43                     ls.get() ; 
44                 } 
45             } 
46         } 
47     } 
48     public static void main(String args[]){ 
49         ThreadDeadLock t1 = new ThreadDeadLock() ;      // 控制張三 
50         ThreadDeadLock t2 = new ThreadDeadLock() ;      // 控制李四 
51         t1.flag = true ; 
52         t2.flag = false ; 
53         Thread thA = new Thread(t1) ; 
54         Thread thB = new Thread(t2) ; 
55         thA.start() ; 
56         thB.start() ; 
57     } 
58 };

程序運行結果:


以下代碼不再執行,程序進入死鎖狀態。

總結

至此關於多線程一些基本操作就介紹完了,鑒於筆者經驗有限,如果有什么不足和缺漏的地方,歡迎相互交流學習,感謝大家!


我有一個微信公眾號,經常會分享一些Java技術相關的干貨;如果你喜歡我的分享,可以用微信搜索“Java團長”或者“javatuanzhang”關注。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM