国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > Java > 正文

Java 多線程學習詳細總結

2019-11-26 13:53:10
字體:
來源:轉載
供稿:網友

目錄(?)[-]

一擴展javalangThread類
二實現javalangRunnable接口
三Thread和Runnable的區別
四線程狀態轉換
五線程調度
六常用函數說明
使用方式
為什么要用join方法
七常見線程名詞解釋
八線程同步
九線程數據傳遞 

    本文主要講了java中多線程的使用方法、線程同步、線程數據傳遞、線程狀態及相應的一些線程函數用法、概述等。

首先講一下進程和線程的區別:

  進程:每個進程都有獨立的代碼和數據空間(進程上下文),進程間的切換會有較大的開銷,一個進程包含1--n個線程。

  線程:同一類線程共享代碼和數據空間,每個線程有獨立的運行棧和程序計數器(PC),線程切換開銷小。

  線程和進程一樣分為五個階段:創建、就緒、運行、阻塞、終止。

  多進程是指操作系統能同時運行多個任務(程序)。

  多線程是指在同一程序中有多個順序流在執行。

在java中要想實現多線程,有兩種手段,一種是繼續Thread類,另外一種是實現Runable接口。

一、擴展java.lang.Thread類

package com.multithread.learning;/** *@functon 多線程學習 *@author 林炳文 *@time 2015.3.9 */class Thread1 extends Thread{ private String name;  public Thread1(String name) {    this.name=name;  } public void run() {    for (int i = 0; i < 5; i++) {      System.out.println(name + "運行 : " + i);      try {        sleep((int) Math.random() * 10);      } catch (InterruptedException e) {        e.printStackTrace();      }    }     }}public class Main { public static void main(String[] args) { Thread1 mTh1=new Thread1("A"); Thread1 mTh2=new Thread1("B"); mTh1.start(); mTh2.start(); }}

輸出:

A運行  :  0
B運行  :  0
A運行  :  1
A運行  :  2
A運行  :  3
A運行  :  4
B運行  :  1
B運行  :  2
B運行  :  3
B運行  :  4

再運行一下:

A運行  :  0
B運行  :  0
B運行  :  1
B運行  :  2
B運行  :  3
B運行  :  4
A運行  :  1
A運行  :  2
A運行  :  3
A運行  :  4

說明:

程序啟動運行main時候,java虛擬機啟動一個進程,主線程main在main()調用時候被創建。隨著調用MitiSay的兩個對象的start方法,另外兩個線程也啟動了,這樣,整個應用就在多線程下運行。

 注意:start()方法的調用后并不是立即執行多線程代碼,而是使得該線程變為可運行態(Runnable),什么時候運行是由操作系統決定的。

從程序運行的結果可以發現,多線程程序是亂序執行。因此,只有亂序執行的代碼才有必要設計為多線程。
Thread.sleep()方法調用目的是不讓當前線程獨自霸占該進程所獲取的CPU資源,以留出一定時間給其他線程執行的機會。

實際上所有的多線程代碼執行順序都是不確定的,每次執行的結果都是隨機的。

但是start方法重復調用的話,會出現java.lang.IllegalThreadStateException異常。

 Thread1 mTh1=new Thread1("A"); Thread1 mTh2=mTh1; mTh1.start(); mTh2.start();

輸出:

Exception in thread "main" java.lang.IllegalThreadStateException
    at java.lang.Thread.start(Unknown Source)
    at com.multithread.learning.Main.main(Main.java:31)

A運行  :  0
A運行  :  1
A運行  :  2
A運行  :  3
A運行  :  4

二、實現java.lang.Runnable接口

/** *@functon 多線程學習 *@author 林炳文 *@time 2015.3.9 */package com.multithread.runnable;class Thread2 implements Runnable{ private String name; public Thread2(String name) { this.name=name; } @Override public void run() {  for (int i = 0; i < 5; i++) {       System.out.println(name + "運行 : " + i);       try {       Thread.sleep((int) Math.random() * 10);       } catch (InterruptedException e) {         e.printStackTrace();       }     }  } }public class Main { public static void main(String[] args) { new Thread(new Thread2("C")).start(); new Thread(new Thread2("D")).start(); }}

輸出:

C運行  :  0
D運行  :  0
D運行  :  1
C運行  :  1
D運行  :  2
C運行  :  2
D運行  :  3
C運行  :  3
D運行  :  4
C運行  :  4

說明:

Thread2類通過實現Runnable接口,使得該類有了多線程類的特征。run()方法是多線程程序的一個約定。所有的多線程代碼都在run方法里面。Thread類實際上也是實現了Runnable接口的類。

在啟動的多線程的時候,需要先通過Thread類的構造方法Thread(Runnable target) 構造出對象,然后調用Thread對象的start()方法來運行多線程代碼。

實際上所有的多線程代碼都是通過運行Thread的start()方法來運行的。因此,不管是擴展Thread類還是實現Runnable接口來實現多線程,最終還是通過Thread的對象的API來控制線程的,熟悉Thread類的API是進行多線程編程的基礎。

三、Thread和Runnable的區別

如果一個類繼承Thread,則不適合資源共享。但是如果實現了Runable接口的話,則很容易的實現資源共享。

package com.multithread.learning;/** *@functon 多線程學習,繼承Thread,資源不能共享 *@author 林炳文 *@time 2015.3.9 */class Thread1 extends Thread{ private int count=5; private String name;  public Thread1(String name) {    this.name=name;  } public void run() {    for (int i = 0; i < 5; i++) {      System.out.println(name + "運行 count= " + count--);      try {        sleep((int) Math.random() * 10);      } catch (InterruptedException e) {        e.printStackTrace();      }    }     }}public class Main { public static void main(String[] args) { Thread1 mTh1=new Thread1("A"); Thread1 mTh2=new Thread1("B"); mTh1.start(); mTh2.start(); }}

輸出:

B運行  count= 5
A運行  count= 5
B運行  count= 4
B運行  count= 3
B運行  count= 2
B運行  count= 1
A運行  count= 4
A運行  count= 3
A運行  count= 2
A運行  count= 1

從上面可以看出,不同的線程之間count是不同的,這對于賣票系統來說就會有很大的問題,當然,這里可以用同步來作。這里我們用Runnable來做下看看

/** *@functon 多線程學習 繼承runnable,資源能共享 *@author 林炳文 *@time 2015.3.9 */package com.multithread.runnable;class Thread2 implements Runnable{  private int count=15; @Override public void run() {  for (int i = 0; i < 5; i++) {   System.out.println(Thread.currentThread().getName() + "運行 count= " + count--);       try {       Thread.sleep((int) Math.random() * 10);       } catch (InterruptedException e) {         e.printStackTrace();       }     }  } }public class Main { public static void main(String[] args) {  Thread2 my = new Thread2();     new Thread(my, "C").start();//同一個mt,但是在Thread中就不可以,如果用同一個實例化對象mt,就會出現異常       new Thread(my, "D").start();     new Thread(my, "E").start(); }}

輸出:

C運行  count= 15
D運行  count= 14
E運行  count= 13
D運行  count= 12
D運行  count= 10
D運行  count= 9
D運行  count= 8
C運行  count= 11
E運行  count= 12
C運行  count= 7
E運行  count= 6
C運行  count= 5
E運行  count= 4
C運行  count= 3
E運行  count= 2

這里要注意每個線程都是用同一個實例化對象,如果不是同一個,效果就和上面的一樣了!

總結:

實現Runnable接口比繼承Thread類所具有的優勢:

1):適合多個相同的程序代碼的線程去處理同一個資源

2):可以避免java中的單繼承的限制

3):增加程序的健壯性,代碼可以被多個線程共享,代碼和數據獨立

提醒一下大家:main方法其實也是一個線程。在java中所以的線程都是同時啟動的,至于什么時候,哪個先執行,完全看誰先得到CPU的資源。

在java中,每次程序運行至少啟動2個線程。一個是main線程,一個是垃圾收集線程。因為每當使用java命令執行一個類的時候,實際上都會啟動一個JVM,每一個jVM實習在就是在操作系統中啟動了一個進程。

四、線程狀態轉換

1、新建狀態(New):新創建了一個線程對象。

2、就緒狀態(Runnable):線程對象創建后,其他線程調用了該對象的start()方法。該狀態的線程位于可運行線程池中,變得可運行,等待獲取CPU的使用權。

3、運行狀態(Running):就緒狀態的線程獲取了CPU,執行程序代碼。

4、阻塞狀態(Blocked):阻塞狀態是線程因為某種原因放棄CPU使用權,暫時停止運行。直到線程進入就緒狀態,才有機會轉到運行狀態。阻塞的情況分三種:

(一)、等待阻塞:運行的線程執行wait()方法,JVM會把該線程放入等待池中。
(二)、同步阻塞:運行的線程在獲取對象的同步鎖時,若該同步鎖被別的線程占用,則JVM會把該線程放入鎖池中。
(三)、其他阻塞:運行的線程執行sleep()或join()方法,或者發出了I/O請求時,JVM會把該線程置為阻塞狀態。當sleep()狀態超時、join()等待線程終止或者超時、或者I/O處理完畢時,線程重新轉入就緒狀態。

5、死亡狀態(Dead):線程執行完了或者因異常退出了run()方法,該線程結束生命周期。

五、線程調度

線程的調度

1、調整線程優先級:Java線程有優先級,優先級高的線程會獲得較多的運行機會。 

Java線程的優先級用整數表示,取值范圍是1~10,Thread類有以下三個靜態常量:
static int MAX_PRIORITY
          線程可以具有的最高優先級,取值為10。
static int MIN_PRIORITY
          線程可以具有的最低優先級,取值為1。
static int NORM_PRIORITY
          分配給線程的默認優先級,取值為5。 

Thread類的setPriority()和getPriority()方法分別用來設置和獲取線程的優先級。

每個線程都有默認的優先級。主線程的默認優先級為Thread.NORM_PRIORITY。
線程的優先級有繼承關系,比如A線程中創建了B線程,那么B將和A具有相同的優先級。
JVM提供了10個線程優先級,但與常見的操作系統都不能很好的映射。如果希望程序能移植到各個操作系統中,應該僅僅使用Thread類有以下三個靜態常量作為優先級,這樣能保證同樣的優先級采用了同樣的調度方式。

2、線程睡眠:Thread.sleep(long millis)方法,使線程轉到阻塞狀態。millis參數設定睡眠的時間,以毫秒為單位。當睡眠結束后,就轉為就緒(Runnable)狀態。sleep()平臺移植性好。

3、線程等待:Object類中的wait()方法,導致當前的線程等待,直到其他線程調用此對象的 notify() 方法或 notifyAll() 喚醒方法。這個兩個喚醒方法也是Object類中的方法,行為等價于調用 wait(0) 一樣。 

4、線程讓步:Thread.yield() 方法,暫停當前正在執行的線程對象,把執行機會讓給相同或者更高優先級的線程。

 5、線程加入:join()方法,等待其他線程終止。在當前線程中調用另一個線程的join()方法,則當前線程轉入阻塞狀態,直到另一個進程運行結束,當前線程再由阻塞轉為就緒狀態。 

6、線程喚醒:Object類中的notify()方法,喚醒在此對象監視器上等待的單個線程。如果所有線程都在此對象上等待,則會選擇喚醒其中一個線程。選擇是任意性的,并在對實現做出決定時發生。線程通過調用其中一個 wait 方法,在對象的監視器上等待。 直到當前的線程放棄此對象上的鎖定,才能繼續執行被喚醒的線程。被喚醒的線程將以常規方式與在該對象上主動同步的其他所有線程進行競爭;例如,喚醒的線程在作為鎖定此對象的下一個線程方面沒有可靠的特權或劣勢。類似的方法還有一個notifyAll(),喚醒在此對象監視器上等待的所有線程。
 注意:Thread中suspend()和resume()兩個方法在JDK1.5中已經廢除,不再介紹。因為有死鎖傾向。

六、常用函數說明

①sleep(long millis): 在指定的毫秒數內讓當前正在執行的線程休眠(暫停執行)

②join():指等待t線程終止。

使用方式。

join是Thread類的一個方法,啟動線程后直接調用,即join()的作用是:“等待該線程終止”,這里需要理解的就是該線程是指的主線程等待子線程的終止。也就是在子線程調用了join()方法后面的代碼,只有等到子線程結束了才能執行。

Thread t = new AThread(); t.start(); t.join();

為什么要用join()方法

在很多情況下,主線程生成并起動了子線程,如果子線程里要進行大量的耗時的運算,主線程往往將于子線程之前結束,但是如果主線程處理完其他的事務后,需要用到子線程的處理結果,也就是主線程需要等待子線程執行完成之后再結束,這個時候就要用到join()方法了。

不加join。/** *@functon 多線程學習,join *@author 林炳文 *@time 2015.3.9 */package com.multithread.join;class Thread1 extends Thread{ private String name;  public Thread1(String name) {   super(name);    this.name=name;  } public void run() { System.out.println(Thread.currentThread().getName() + " 線程運行開始!");    for (int i = 0; i < 5; i++) {      System.out.println("子線程"+name + "運行 : " + i);      try {        sleep((int) Math.random() * 10);      } catch (InterruptedException e) {        e.printStackTrace();      }    }    System.out.println(Thread.currentThread().getName() + " 線程運行結束!"); }}public class Main { public static void main(String[] args) { System.out.println(Thread.currentThread().getName()+"主線程運行開始!"); Thread1 mTh1=new Thread1("A"); Thread1 mTh2=new Thread1("B"); mTh1.start(); mTh2.start(); System.out.println(Thread.currentThread().getName()+ "主線程運行結束!"); }}

輸出結果:

main主線程運行開始!
main主線程運行結束!
B 線程運行開始!
子線程B運行 : 0
A 線程運行開始!
子線程A運行 : 0
子線程B運行 : 1
子線程A運行 : 1
子線程A運行 : 2
子線程A運行 : 3
子線程A運行 : 4
A 線程運行結束!
子線程B運行 : 2
子線程B運行 : 3
子線程B運行 : 4
B 線程運行結束!
發現主線程比子線程早結束

加join

public class Main { public static void main(String[] args) { System.out.println(Thread.currentThread().getName()+"主線程運行開始!"); Thread1 mTh1=new Thread1("A"); Thread1 mTh2=new Thread1("B"); mTh1.start(); mTh2.start(); try {  mTh1.join(); } catch (InterruptedException e) {  e.printStackTrace(); } try {  mTh2.join(); } catch (InterruptedException e) {  e.printStackTrace(); } System.out.println(Thread.currentThread().getName()+ "主線程運行結束!"); }}

運行結果:

main主線程運行開始!
A 線程運行開始!
子線程A運行 : 0
B 線程運行開始!
子線程B運行 : 0
子線程A運行 : 1
子線程B運行 : 1
子線程A運行 : 2
子線程B運行 : 2
子線程A運行 : 3
子線程B運行 : 3
子線程A運行 : 4
子線程B運行 : 4
A 線程運行結束!

主線程一定會等子線程都結束了才結束

③yield():暫停當前正在執行的線程對象,并執行其他線程。
        Thread.yield()方法作用是:暫停當前正在執行的線程對象,并執行其他線程。
         yield()應該做的是讓當前運行線程回到可運行狀態,以允許具有相同優先級的其他線程獲得運行機會。因此,使用yield()的目的是讓相同優先級的線程之間能適當的輪轉執行。但是,實際中無法保證yield()達到讓步目的,因為讓步的線程還有可能被線程調度程序再次選中。 

結論:yield()從未導致線程轉到等待/睡眠/阻塞狀態。在大多數情況下,yield()將導致線程從運行狀態轉到可運行狀態,但有可能沒有效果。可看上面的圖。

/** *@functon 多線程學習 yield *@author 林炳文 *@time 2015.3.9 */package com.multithread.yield;class ThreadYield extends Thread{  public ThreadYield(String name) {    super(name);  }   @Override  public void run() {    for (int i = 1; i <= 50; i++) {      System.out.println("" + this.getName() + "-----" + i);      // 當i為30時,該線程就會把CPU時間讓掉,讓其他或者自己的線程執行(也就是誰先搶到誰執行)      if (i ==30) {        this.yield();      }    } }}public class Main { public static void main(String[] args) {  ThreadYield yt1 = new ThreadYield("張三");   ThreadYield yt2 = new ThreadYield("李四");    yt1.start();    yt2.start(); }}

運行結果:

第一種情況:李四(線程)當執行到30時會CPU時間讓掉,這時張三(線程)搶到CPU時間并執行。

第二種情況:李四(線程)當執行到30時會CPU時間讓掉,這時李四(線程)搶到CPU時間并執行。

sleep()和yield()的區別

        sleep()和yield()的區別):sleep()使當前線程進入停滯狀態,所以執行sleep()的線程在指定的時間內肯定不會被執行;yield()只是使當前線程重新回到可執行狀態,所以執行yield()的線程有可能在進入到可執行狀態后馬上又被執行。

        sleep 方法使當前運行中的線程睡眼一段時間,進入不可運行狀態,這段時間的長短是由程序設定的,yield 方法使當前線程讓出 CPU 占有權,但讓出的時間是不可設定的。實際上,yield()方法對應了如下操作:先檢測當前是否有相同優先級的線程處于同可運行狀態,如有,則把 CPU  的占有權交給此線程,否則,繼續運行原來的線程。所以yield()方法稱為“退讓”,它把運行機會讓給了同等優先級的其他線程

       另外,sleep 方法允許較低優先級的線程獲得運行機會,但 yield()  方法執行時,當前線程仍處在可運行狀態,所以,不可能讓出較低優先級的線程些時獲得 CPU 占有權。在一個運行系統中,如果較高優先級的線程沒有調用 sleep 方法,又沒有受到 I/O 阻塞,那么,較低優先級線程只能等待所有較高優先級的線程運行結束,才有機會運行。

④setPriority(): 更改線程的優先級。

    MIN_PRIORITY = 1
       NORM_PRIORITY = 5
           MAX_PRIORITY = 10

用法:

Thread4 t1 = new Thread4("t1");
Thread4 t2 = new Thread4("t2");
t1.setPriority(Thread.MAX_PRIORITY);
t2.setPriority(Thread.MIN_PRIORITY);

⑤interrupt():中斷某個線程,這種結束方式比較粗暴,如果t線程打開了某個資源還沒來得及關閉也就是run方法還沒有執行完就強制結束線程,會導致資源無法關閉

  要想結束進程最好的辦法就是用sleep()函數的例子程序里那樣,在線程類里面用以個boolean型變量來控制run()方法什么時候結束,run()方法一結束,該線程也就結束了。

⑥wait()

Obj.wait(),與Obj.notify()必須要與synchronized(Obj)一起使用,也就是wait,與notify是針對已經獲取了Obj鎖進行操作,從語法角度來說就是Obj.wait(),Obj.notify必須在synchronized(Obj){...}語句塊內。從功能上來說wait就是說線程在獲取對象鎖后,主動釋放對象鎖,同時本線程休眠。直到有其它線程調用對象的notify()喚醒該線程,才能繼續獲取對象鎖,并繼續執行。相應的notify()就是對對象鎖的喚醒操作。但有一點需要注意的是notify()調用后,并不是馬上就釋放對象鎖的,而是在相應的synchronized(){}語句塊執行結束,自動釋放鎖后,JVM會在wait()對象鎖的線程中隨機選取一線程,賦予其對象鎖,喚醒線程,繼續執行。這樣就提供了在線程間同步、喚醒的操作。Thread.sleep()與Object.wait()二者都可以暫停當前線程,釋放CPU控制權,主要的區別在于Object.wait()在釋放CPU同時,釋放了對象鎖的控制。

    單單在概念上理解清楚了還不夠,需要在實際的例子中進行測試才能更好的理解。對Object.wait(),Object.notify()的應用最經典的例子,應該是三線程打印ABC的問題了吧,這是一道比較經典的面試題,題目要求如下:

    建立三個線程,A線程打印10次A,B線程打印10次B,C線程打印10次C,要求線程同時運行,交替打印10次ABC。這個問題用Object的wait(),notify()就可以很方便的解決。代碼如下:

/** * wait用法 * @author DreamSea  * @time 2015.3.9  */package com.multithread.wait;public class MyThreadPrinter2 implements Runnable {      private String name;    private Object prev;    private Object self;     private MyThreadPrinter2(String name, Object prev, Object self) {      this.name = name;      this.prev = prev;      this.self = self;    }     @Override   public void run() {      int count = 10;      while (count > 0) {        synchronized (prev) {          synchronized (self) {            System.out.print(name);            count--;                     self.notify();          }          try {            prev.wait();          } catch (InterruptedException e) {            e.printStackTrace();          }        }       }    }     public static void main(String[] args) throws Exception {      Object a = new Object();      Object b = new Object();      Object c = new Object();      MyThreadPrinter2 pa = new MyThreadPrinter2("A", c, a);      MyThreadPrinter2 pb = new MyThreadPrinter2("B", a, b);      MyThreadPrinter2 pc = new MyThreadPrinter2("C", b, c);                  new Thread(pa).start();    Thread.sleep(100); //確保按順序A、B、C執行    new Thread(pb).start();    Thread.sleep(100);     new Thread(pc).start();      Thread.sleep(100);     }  } 

輸出結果:

ABCABCABCABCABCABCABCABCABCABC

     先來解釋一下其整體思路,從大的方向上來講,該問題為三線程間的同步喚醒操作,主要的目的就是ThreadA->ThreadB->ThreadC->ThreadA循環執行三個線程。為了控制線程執行的順序,那么就必須要確定喚醒、等待的順序,所以每一個線程必須同時持有兩個對象鎖,才能繼續執行。一個對象鎖是prev,就是前一個線程所持有的對象鎖。還有一個就是自身對象鎖。主要的思想就是,為了控制執行的順序,必須要先持有prev鎖,也就前一個線程要釋放自身對象鎖,再去申請自身對象鎖,兩者兼備時打印,之后首先調用self.notify()釋放自身對象鎖,喚醒下一個等待線程,再調用prev.wait()釋放prev對象鎖,終止當前線程,等待循環結束后再次被喚醒。運行上述代碼,可以發現三個線程循環打印ABC,共10次。程序運行的主要過程就是A線程最先運行,持有C,A對象鎖,后釋放A,C鎖,喚醒B。線程B等待A鎖,再申請B鎖,后打印B,再釋放B,A鎖,喚醒C,線程C等待B鎖,再申請C鎖,后打印C,再釋放C,B鎖,喚醒A。看起來似乎沒什么問題,但如果你仔細想一下,就會發現有問題,就是初始條件,三個線程按照A,B,C的順序來啟動,按照前面的思考,A喚醒B,B喚醒C,C再喚醒A。但是這種假設依賴于JVM中線程調度、執行的順序。

   wait和sleep區別

共同點:

1. 他們都是在多線程的環境下,都可以在程序的調用處阻塞指定的毫秒數,并返回。
2. wait()和sleep()都可以通過interrupt()方法 打斷線程的暫停狀態 ,從而使線程立刻拋出InterruptedException。
   如果線程A希望立即結束線程B,則可以對線程B對應的Thread實例調用interrupt方法。如果此刻線程B正在wait/sleep /join,則線程B會立刻拋出InterruptedException,在catch() {} 中直接return即可安全地結束線程。
   需要注意的是,InterruptedException是線程自己從內部拋出的,并不是interrupt()方法拋出的。對某一線程調用 interrupt()時,如果該線程正在執行普通的代碼,那么該線程根本就不會拋出InterruptedException。但是,一旦該線程進入到 wait()/sleep()/join()后,就會立刻拋出InterruptedException 。

不同點:

1. Thread類的方法:sleep(),yield()等
   Object的方法:wait()和notify()等
2. 每個對象都有一個鎖來控制同步訪問。Synchronized關鍵字可以和對象的鎖交互,來實現線程的同步。
   sleep方法沒有釋放鎖,而wait方法釋放了鎖,使得其他線程可以使用同步控制塊或者方法。
3. wait,notify和notifyAll只能在同步控制方法或者同步控制塊里面使用,而sleep可以在任何地方使用
4. sleep必須捕獲異常,而wait,notify和notifyAll不需要捕獲異常
所以sleep()和wait()方法的最大區別是:
    sleep()睡眠時,保持對象鎖,仍然占有該鎖;
    而wait()睡眠時,釋放對象鎖。
  但是wait()和sleep()都可以通過interrupt()方法打斷線程的暫停狀態,從而使線程立刻拋出InterruptedException(但不建議使用該方法)。

sleep()方法

sleep()使當前線程進入停滯狀態(阻塞當前線程),讓出CUP的使用、目的是不讓當前線程獨自霸占該進程所獲的CPU資源,以留一定時間給其他線程執行的機會;
   sleep()是Thread類的Static(靜態)的方法;因此他不能改變對象的機鎖,所以當在一個Synchronized塊中調用Sleep()方法是,線程雖然休眠了,但是對象的機鎖并木有被釋放,其他線程無法訪問這個對象(即使睡著也持有對象鎖)。
  在sleep()休眠時間期滿后,該線程不一定會立即執行,這是因為其它線程可能正在運行而且沒有被調度為放棄執行,除非此線程具有更高的優先級。

wait()方法

wait()方法是Object類里的方法;當一個線程執行到wait()方法時,它就進入到一個和該對象相關的等待池中,同時失去(釋放)了對象的機鎖(暫時失去機鎖,wait(long timeout)超時時間到后還需要返還對象鎖);其他線程可以訪問;
  wait()使用notify或者notifyAlll或者指定睡眠時間來喚醒當前等待池中的線程。
  wiat()必須放在synchronized block中,否則會在program runtime時扔出”java.lang.IllegalMonitorStateException“異常。

七、常見線程名詞解釋

主線程:JVM調用程序main()所產生的線程。
當前線程:這個是容易混淆的概念。一般指通過Thread.currentThread()來獲取的進程。
后臺線程:指為其他線程提供服務的線程,也稱為守護線程。JVM的垃圾回收線程就是一個后臺線程。用戶線程和守護線程的區別在于,是否等待主線程依賴于主線程結束而結束
前臺線程:是指接受后臺線程服務的線程,其實前臺后臺線程是聯系在一起,就像傀儡和幕后操縱者一樣的關系。傀儡是前臺線程、幕后操縱者是后臺線程。由前臺線程創建的線程默認也是前臺線程。可以通過isDaemon()和setDaemon()方法來判斷和設置一個線程是否為后臺線程。
線程類的一些常用方法:

  sleep(): 強迫一個線程睡眠N毫秒。
  isAlive(): 判斷一個線程是否存活。
  join(): 等待線程終止。
  activeCount(): 程序中活躍的線程數。
  enumerate(): 枚舉程序中的線程。
     currentThread(): 得到當前線程。
  isDaemon(): 一個線程是否為守護線程。
  setDaemon(): 設置一個線程為守護線程。(用戶線程和守護線程的區別在于,是否等待主線程依賴于主線程結束而結束)
  setName(): 為線程設置一個名稱。
  wait(): 強迫一個線程等待。
  notify(): 通知一個線程繼續運行。
  setPriority(): 設置一個線程的優先級。

八、線程同步

1、synchronized關鍵字的作用域有二種:

1)是某個對象實例內,synchronized aMethod(){}可以防止多個線程同時訪問這個對象的synchronized方法(如果一個對象有多個synchronized方法,只要一個線程訪問了其中的一個synchronized方法,其它線程不能同時訪問這個對象中任何一個synchronized方法)。這時,不同的對象實例的synchronized方法是不相干擾的。也就是說,其它線程照樣可以同時訪問相同類的另一個對象實例中的synchronized方法;
2)是某個類的范圍,synchronized static aStaticMethod{}防止多個線程同時訪問這個類中的synchronized static 方法。它可以對類的所有對象實例起作用。

2、除了方法前用synchronized關鍵字,synchronized關鍵字還可以用于方法中的某個區塊中,表示只對這個區塊的資源實行互斥訪問。用法是: synchronized(this){/*區塊*/},它的作用域是當前對象;

3、synchronized關鍵字是不能繼承的,也就是說,基類的方法synchronized f(){} 在繼承類中并不自動是synchronized f(){},而是變成了f(){}。繼承類需要你顯式的指定它的某個方法為synchronized方法;

Java對多線程的支持與同步機制深受大家的喜愛,似乎看起來使用了synchronized關鍵字就可以輕松地解決多線程共享數據同步問題。到底如何?

主站蜘蛛池模板: 蓬安县| 略阳县| 青海省| 汉中市| 胶南市| 巴彦淖尔市| 鲁甸县| 南召县| 宁乡县| 伊川县| 宁化县| 穆棱市| 汉中市| 岳西县| 察雅县| 沅陵县| 郯城县| 璧山县| 郎溪县| 蒙城县| 聊城市| 厦门市| 林口县| 定襄县| 苍梧县| 吉隆县| 攀枝花市| 瓮安县| 千阳县| 罗平县| 江都市| 湾仔区| 揭东县| 德兴市| 新和县| 大田县| 垣曲县| 农安县| 浑源县| 临颍县| 靖宇县|