成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Java并發(fā)編程——線程基礎(chǔ)查漏補(bǔ)缺

luqiuwen / 1472人閱讀

摘要:告訴當(dāng)前執(zhí)行的線程為線程池中其他具有相同優(yōu)先級的線程提供機(jī)會。不能保證會立即使當(dāng)前正在執(zhí)行的線程處于可運行狀態(tài)。當(dāng)達(dá)到超時時間時,主線程和是同樣可能的執(zhí)行者候選。下一篇并發(fā)編程線程安全性深層原因

Thread

使用Java的同學(xué)對Thread應(yīng)該不陌生了,線程的創(chuàng)建和啟動等這里就不講了,這篇主要講幾個容易被忽視的方法以及線程狀態(tài)遷移。

wait/notify/notifyAll

首先我們要明白這三個方法是定義在Object類中,他們起到的作用就是允許線程就資源的鎖定狀態(tài)進(jìn)行通信。這里所說的資源一般就是指的我們常說的共享對象了,也就是說針對共享對象的鎖定狀態(tài)可以通過wait/notify/notifyAll來進(jìn)行通信。我們先看下如何使用的,并對相應(yīng)原理進(jìn)行展開。

wait

wait方法告訴調(diào)用線程放棄鎖定并進(jìn)入休眠狀態(tài),直到其他某個線程進(jìn)入同一個監(jiān)視器(monitor)并調(diào)用notify方法。wait方法在等待之前釋放鎖,并在wait方法返回之前重新獲取鎖。wait方法實際上和同步鎖緊密集成,補(bǔ)充同步機(jī)制無法直接實現(xiàn)的功能。
需要注意到wait方法在jdk源碼中是final并且是native的本地方法,我們無法去覆蓋此方法。
調(diào)用wait一般的方式如下:

synchronized(lockObject) {
    while(!condition) {
        lockObject.wait();
    }
    // 這里進(jìn)行相應(yīng)處理;
}

注意這里使用while進(jìn)行條件判斷而沒有使用if進(jìn)行條件判斷,原因是這里有個很重要的點容易被忽視,下面來自官方的建議:

應(yīng)該在循環(huán)中檢查等待條件,原因是處于等待狀態(tài)的線程可能會收到錯誤的警報和偽喚醒,如果不在循環(huán)條件中等待,程序就會在沒有滿足結(jié)束條件的情況下退出。
notify

notify方法喚醒了同一個對象上調(diào)用wait的線程。這里要注意notify并沒有放棄對資源的鎖定,他告訴等待的線程可以喚醒,但是作用在notify上synchronized同步塊完成之前,實際上是不會放棄鎖。因此,如果通知線程在同步塊內(nèi),調(diào)用notify方法后,需要在進(jìn)行10s的其他操作,那么等待的線程將會再至少等待10s。
notify一般的使用方式如下:

synchronized(lockObject) {
    // 確定條件
    lockObject.notify();
    // 如果需要可以加任意代碼
}
notifyAll

notifyAll會喚醒在同一個對象上調(diào)用wait方法的所有線程。在大多數(shù)情況下優(yōu)先級最高的線程將被執(zhí)行,但是也是無法完全保證會是這樣。其他的與notify相同。

使用例子

下面的代碼示例實現(xiàn)了隊列空和滿時線程阻塞已經(jīng)非空非滿時的通知:

生產(chǎn)者:

class Producer implements Runnable {
    private final List taskQueue;
    private final int MAX_CAPACITY;

    public Producer(List sharedQueue, int size) {
        this.taskQueue = sharedQueue;
        this.MAX_CAPACITY = size;
    }

    @Override
    public void run() {
        int counter = 0;
        while (true) {
            try {
                produce(counter++);
            } catch (InterruptedException ex) {
                ex.printStackTrace();
            }
        }
    }

    private void produce(int i) throws InterruptedException {
        synchronized (taskQueue) {
            while (taskQueue.size() == MAX_CAPACITY) {
                System.out.println("隊列已滿,線程" + Thread.currentThread().getName() + "進(jìn)入等待,隊列長度:" + taskQueue.size());
                taskQueue.wait();
            }

            Thread.sleep(1000);
            taskQueue.add(i);
            System.out.println("生產(chǎn):" + i);
            taskQueue.notifyAll();
        }
    }
}

消費者:

class Consumer implements Runnable {
    private final List taskQueue;

    public Consumer(List sharedQueue) {
        this.taskQueue = sharedQueue;
    }

    @Override
    public void run() {
        while (true) {
            try {
                consume();
            } catch (InterruptedException ex) {
                ex.printStackTrace();
            }
        }
    }

    private void consume() throws InterruptedException {
        synchronized (taskQueue) {
            while (taskQueue.isEmpty()) {
                System.out.println("隊列已空,線程" + Thread.currentThread().getName() + "進(jìn)入等待,隊列長度:" + taskQueue.size());
                taskQueue.wait();
            }
            Thread.sleep(1000);
            int i = (Integer) taskQueue.remove(0);
            System.out.println("消費:" + i);
            taskQueue.notifyAll();
        }
    }
}

測試代碼:

public class ProducerConsumerExampleWithWaitAndNotify {
    public static void main(String[] args) {
        List taskQueue = new ArrayList<>();
        int MAX_CAPACITY = 5;
        Thread tProducer = new Thread(new Producer(taskQueue, MAX_CAPACITY), "Producer");
        Thread tConsumer = new Thread(new Consumer(taskQueue), "Consumer");
        tProducer.start();
        tConsumer.start();
    }
}

部分輸出如下:

生產(chǎn):0
生產(chǎn):1
生產(chǎn):2
生產(chǎn):3
生產(chǎn):4
隊列已滿,線程Producer進(jìn)入等待,隊列長度:5
消費:0
消費:1
消費:2
消費:3
消費:4
隊列已空,線程Consumer進(jìn)入等待,隊列長度:0
yield/join yield

從字面意思理解yield可以是謙讓、放棄、屈服、投降的意思。一個要“謙讓”的線程其實是在告訴虛擬機(jī)他愿意讓其他線程安排到他的前面,這表明他沒有說明重要的事情要做了。注意了這只是個提示,并不能保證能起到任何效果。
yield在Thread.java中定義如下:

/**
 * A hint to the scheduler that the current thread is willing to yield
 * its current use of a processor. The scheduler is free to ignore this
 * hint.
 *  * 

Yield is a heuristic attempt to improve relative progression * between threads that would otherwise over-utilise a CPU. Its use * should be combined with detailed profiling and benchmarking to * ensure that it actually has the desired effect. * *

It is rarely appropriate to use this method. It may be useful * for debugging or testing purposes, where it may help to reproduce * bugs due to race conditions. It may also be useful when designing * concurrency control constructs such as the ones in the * {@link java.util.concurrent.locks} package. */ public static native void yield();

從這里面我們總結(jié)出一些重點(有關(guān)線程狀態(tài)后面會講到):

yield方法是一個靜態(tài)的并且是native的方法。

yield告訴當(dāng)前執(zhí)行的線程為線程池中其他具有相同優(yōu)先級的線程提供機(jī)會。

不能保證yield會立即使當(dāng)前正在執(zhí)行的線程處于可運行狀態(tài)。

他只能使得線程從運行狀態(tài)變成可運行狀態(tài),而無法做其他狀態(tài)改變。

yield使用例子:

public class YieldExample {
   public static void main(String[] args) {
      Thread producer = new Producer();
      Thread consumer = new Consumer();
       
      producer.setPriority(Thread.MIN_PRIORITY); // 最低優(yōu)先級
      consumer.setPriority(Thread.MAX_PRIORITY); // 最高優(yōu)先級
       
      producer.start();
      consumer.start();
   }
}
 
class Producer extends Thread {
   public void run() {
      for (int i = 0; i < 5; i++) {
         System.out.println("生產(chǎn)者 : 生產(chǎn) " + i);
         Thread.yield();
      }
   }
}
 
class Consumer extends Thread {
   public void run() {
      for (int i = 0; i < 5; i++) {
         System.out.println("消費者 : 消費 " + i);
         Thread.yield();
      }
   }
}

當(dāng)注釋兩個“Thread.yield();”時輸出:

消費者 : 消費 0
消費者 : 消費 1
消費者 : 消費 2
消費者 : 消費 3
消費者 : 消費 4
生產(chǎn)者 : 生產(chǎn) 0
生產(chǎn)者 : 生產(chǎn) 1
生產(chǎn)者 : 生產(chǎn) 2
生產(chǎn)者 : 生產(chǎn) 3
生產(chǎn)者 : 生產(chǎn) 4

當(dāng)不注釋兩個“Thread.yield();”時輸出:

生產(chǎn)者 : 生產(chǎn) 0
消費者 : 消費 0
生產(chǎn)者 : 生產(chǎn) 1
消費者 : 消費 1
生產(chǎn)者 : 生產(chǎn) 2
消費者 : 消費 2
生產(chǎn)者 : 生產(chǎn) 3
消費者 : 消費 3
生產(chǎn)者 : 生產(chǎn) 4
消費者 : 消費 4
join

join方法用于將線程當(dāng)前執(zhí)行點連接到另一個線程的執(zhí)行結(jié)束,這樣這個線程就不會開始運行直到另一個線程結(jié)束。在Thread實例上調(diào)用join,則當(dāng)前運行的線程將會阻塞,直到這個Thread實例完成執(zhí)行。
簡要摘抄Thread.java源碼中join的定義:

// Waits for this thread to die.
public final void join() throws InterruptedException

join還有可以傳入時間參數(shù)的重載方法,這個可以時join的效果在特定時間后無效。當(dāng)達(dá)到超時時間時,主線程和taskThread是同樣可能的執(zhí)行者候選。但是join和sleep一樣,依賴于OS進(jìn)行計時,不應(yīng)該假定剛好等待指定的時間。
join和sleep一樣也通過InterruptedException來響應(yīng)中斷。
join使用示例:

public class JoinExample {
   public static void main(String[] args) throws InterruptedException {
      Thread t = new Thread(new Runnable() {
          public void run() {
              System.out.println("第一個任務(wù)啟動");
              System.out.println("睡眠2s");
              try {
                  Thread.sleep(2000);
              } catch (InterruptedException e) {
                  e.printStackTrace();
              }
              System.out.println("第一個任務(wù)完成");
          }
      });
      Thread t1 = new Thread(new Runnable() {
          public void run() {
              System.out.println("第二個任務(wù)完成");
          }
      });
      t.start();
      t.join();
      t1.start();
   }
}

輸出結(jié)果:

第一個任務(wù)啟動
睡眠2s
第一個任務(wù)完成
第二個任務(wù)完成
join原理分析

join在Thread.java中有三個重載方法:

public final void join() throws InterruptedException

public final synchronized void join(long millis) throws InterruptedException

public final synchronized void join(long millis, int nanos) throws InterruptedException

查看源碼可以得知最終的實現(xiàn)核心部分都在join(long millis)中,我們來分析下這個方法源碼:

public final synchronized void join(long millis)
throws InterruptedException {
    long base = System.currentTimeMillis();
    long now = 0;

    if (millis < 0) {
        throw new IllegalArgumentException("timeout value is negative");
    }

    if (millis == 0) {
        while (isAlive()) {
            wait(0);
        }
    } else {
        while (isAlive()) {
            long delay = millis - now;
            if (delay <= 0) {
                break;
            }
            wait(delay);
            now = System.currentTimeMillis() - base;
        }
    }
}

首先可以看到這個方法是使用synchronized修飾的同步方法,從這個方法的源碼可以看出join的核心就是使用wait來實現(xiàn)的,而外部條件就是isAlive(),可以斷定,在非isAlive()時會進(jìn)行notify。

線程狀態(tài)

在Thread.java的源代碼中就體現(xiàn)出了六種狀態(tài):

   /**
     * A thread state.  A thread can be in one of the following states:
     * 
    *
  • {@link #NEW}
    * A thread that has not yet started is in this state. *
  • *
  • {@link #RUNNABLE}
    * A thread executing in the Java virtual machine is in this state. *
  • *
  • {@link #BLOCKED}
    * A thread that is blocked waiting for a monitor lock * is in this state. *
  • *
  • {@link #WAITING}
    * A thread that is waiting indefinitely for another thread to * perform a particular action is in this state. *
  • *
  • {@link #TIMED_WAITING}
    * A thread that is waiting for another thread to perform an action * for up to a specified waiting time is in this state. *
  • *
  • {@link #TERMINATED}
    * A thread that has exited is in this state. *
  • *
* *

* A thread can be in only one state at a given point in time. * These states are virtual machine states which do not reflect * any operating system thread states. * * @since 1.5 * @see #getState */ public enum State { /** * Thread state for a thread which has not yet started. */ NEW, /** * Thread state for a runnable thread. A thread in the runnable * state is executing in the Java virtual machine but it may * be waiting for other resources from the operating system * such as processor. */ RUNNABLE, /** * Thread state for a thread blocked waiting for a monitor lock. * A thread in the blocked state is waiting for a monitor lock * to enter a synchronized block/method or * reenter a synchronized block/method after calling * {@link Object#wait() Object.wait}. */ BLOCKED, /** * Thread state for a waiting thread. * A thread is in the waiting state due to calling one of the * following methods: *

    *
  • {@link Object#wait() Object.wait} with no timeout
  • *
  • {@link #join() Thread.join} with no timeout
  • *
  • {@link LockSupport#park() LockSupport.park}
  • *
* *

A thread in the waiting state is waiting for another thread to * perform a particular action. * * For example, a thread that has called Object.wait() * on an object is waiting for another thread to call * Object.notify() or Object.notifyAll() on * that object. A thread that has called Thread.join() * is waiting for a specified thread to terminate. */ WAITING, /** * Thread state for a waiting thread with a specified waiting time. * A thread is in the timed waiting state due to calling one of * the following methods with a specified positive waiting time: *

    *
  • {@link #sleep Thread.sleep}
  • *
  • {@link Object#wait(long) Object.wait} with timeout
  • *
  • {@link #join(long) Thread.join} with timeout
  • *
  • {@link LockSupport#parkNanos LockSupport.parkNanos}
  • *
  • {@link LockSupport#parkUntil LockSupport.parkUntil}
  • *
*/ TIMED_WAITING, /** * Thread state for a terminated thread. * The thread has completed execution. */ TERMINATED; }

一般我們用如下圖來表示狀態(tài)遷移,注意相關(guān)方法。(注意:其中RUNNING和READY是無法直接獲取的狀態(tài)。)

下一篇:Java并發(fā)編程——線程安全性深層原因

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/72737.html

相關(guān)文章

  • Java并發(fā)編程——線程安全性深層原因

    摘要:線程安全性深層原因這里我們將會從計算機(jī)硬件和編輯器等方面來詳細(xì)了解線程安全產(chǎn)生的深層原因。類似這種不影響單線程語義的亂序執(zhí)行我們稱為指令重排。通過線程安全性深層原因我們能更好的理解這三大性質(zhì)的根本性原因。上一篇并發(fā)編程線程基礎(chǔ)查漏補(bǔ)缺 線程安全性深層原因 這里我們將會從計算機(jī)硬件和編輯器等方面來詳細(xì)了解線程安全產(chǎn)生的深層原因。 緩存一致性問題 CPU內(nèi)存架構(gòu) 隨著CPU的發(fā)展,而因為C...

    Faremax 評論0 收藏0
  • 作為我的的第一門語言,學(xué)習(xí)Java時是什么感受?

    摘要:作為技術(shù)書籍或者視頻,講解一門語言的時候都是從最底層開始講解,底層的基礎(chǔ)有哪些呢首先是整個,讓我們對這門語言先混個臉熟,知道程序的基本結(jié)構(gòu),順帶著還會說一下注釋是什么樣子。 2018年新年剛過,就迷茫了,Java學(xué)不下去了,不知道從哪里學(xué)了。 那么多細(xì)節(jié)的東西,我根本記不住,看完就忘。 剛開始學(xué)習(xí)的時候熱情萬丈,持續(xù)不了幾天就慢慢退去。 作為技術(shù)書籍或者視頻,講解一門語言的時候都是...

    isaced 評論0 收藏0
  • 【面試篇】JS基礎(chǔ)知識查漏補(bǔ)缺

    摘要:因為在頁面加載完成后,引擎維護(hù)著兩個隊列,一個是按頁面順序加載的執(zhí)行隊列,還有一個空閑隊列,使用定時函數(shù)就是將回調(diào)函數(shù)加入到空閑隊列中,故和其他定時器是并發(fā)執(zhí)行的。 1.window.onload和$(document).ready()的區(qū)別: ①執(zhí)行時間:window.onload會在所有元素,包括圖片,引用文件加載完成之后執(zhí)行,而$(document).ready()則會在HTML...

    myeveryheart 評論0 收藏0
  • 【推薦】最新200篇:技術(shù)文章整理

    摘要:作為面試官,我是如何甄別應(yīng)聘者的包裝程度語言和等其他語言的對比分析和主從復(fù)制的原理詳解和持久化的原理是什么面試中經(jīng)常被問到的持久化與恢復(fù)實現(xiàn)故障恢復(fù)自動化詳解哨兵技術(shù)查漏補(bǔ)缺最易錯過的技術(shù)要點大掃盲意外宕機(jī)不難解決,但你真的懂?dāng)?shù)據(jù)恢復(fù)嗎每秒 作為面試官,我是如何甄別應(yīng)聘者的包裝程度Go語言和Java、python等其他語言的對比分析 Redis和MySQL Redis:主從復(fù)制的原理詳...

    BicycleWarrior 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<