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

資訊專欄INFORMATION COLUMN

Java多線程基礎知識(一)

馬龍駒 / 3356人閱讀

摘要:多線程一線程模型實現(xiàn)線程有三種方式使用內核線程實現(xiàn)使用用戶線程實現(xiàn)和使用用戶線程加輕量級進程混合實現(xiàn)。這種輕量級進程與內核線程之間的關系稱為一對一的線程模型。是通知所有等待對象控制權的線程繼續(xù)運行。

Java多線程 一、Java線程模型

實現(xiàn)線程有三種方式:使用內核線程實現(xiàn)、使用用戶線程實現(xiàn)和使用用戶線程加輕量級進程混合實現(xiàn)。內核線程是直接由操作系統(tǒng)內核支持的線程,通過內核完成線程切換,內核通過操縱調度器對線程進行調度,并負責將線程的任務映射到各個處理器上。
程序不會直接使用內核線程,而是去使用內核線程的高級接口-輕量級進程。每個輕量級進程由一個內核線程支持。這種輕量級進程與內核線程之間1:1的關系稱為一對一的線程模型。Sun JDK 的Windows版和Linux版都是使用一對一的線程模型,一條Java線程就映射到一條輕量級進程中。由于內核線程的支持,每個輕量級進程成為一個獨立的調度單元,一個輕量級進程的阻塞不會影響整個進程。但也是因為基于內核線程實現(xiàn),各種線程操作,如創(chuàng)建、析構及同步都要進行系統(tǒng)調用,需要在用戶態(tài)和內核態(tài)中來回切換,調用代價高,其次輕量級進程消耗一定的內核資源,因此一個系統(tǒng)支持輕量級進程的數(shù)量有限。

二、Java線程調度

線程調度是指系統(tǒng)為線程分配處理器使用權的過程,分為協(xié)同式調度和搶占式調度。協(xié)同式調度的多線程系統(tǒng),線程執(zhí)行時間由線程本身控制,線程完成自己的工作之后,主動通知系統(tǒng)切換到另一個線程上。優(yōu)點是實現(xiàn)簡單,切換操作是由線程主動的,對線程可知,沒有線程同步問題。缺點是線程執(zhí)行時間不可控制,如果一個線程阻塞,可能導致整個系統(tǒng)奔潰。搶占式調度的多線程系統(tǒng),每個線程有系統(tǒng)分配執(zhí)行時間,線程的切換不由線程本身決定。(yield可以讓出執(zhí)行時間,但線程本身無法獲取執(zhí)行時間)優(yōu)點是線程執(zhí)行時間系統(tǒng)可控。Java使用的線程調度方式就是搶占式調度。

三、Java線程狀態(tài)

Java線程的6種狀態(tài):

New(新建)

Runnable(可運行)

Blocked(阻塞)

Waiting(等待)

Timed waiting(限時等待)

Terminated(終止)

線程創(chuàng)建成功但尚未啟動就是New;Runable狀態(tài)的線程可能正在執(zhí)行,也可能在等待CPU分配執(zhí)行時間;當線程等待另一個線程通知調度器一個條件時就進入等待狀態(tài),例如Object.wait、Thread.join;當這些方法指定時間參數(shù)時就成了限時等待;當一個線程試圖獲取一個內部的對象鎖,而該鎖被另一線程持有時,該線程進入阻塞狀態(tài);當線程因run方法正常退出而自然死亡,或者因為沒有捕獲的異常死亡都會導致線程進入Terminated狀態(tài)。

四、中斷

Java中斷機制是一種協(xié)作機制,通過中斷并不能直接終止另一個線程,而需要被中斷的線程自己處理中斷。當對一個線程調用interrup方法時,線程的中斷狀態(tài)將被置位。這是每一個線程都具有的boolean標志位。每個線程都應該不時地檢查這個標志,以判斷線程是否被中斷,并及時處理。

public void interrupt() {
    if (this != Thread.currentThread())
        checkAccess();

    synchronized (blockerLock) {
        Interruptible b = blocker;
        if (b != null) {
            interrupt0();           // Just to set the interrupt flag
            b.interrupt(this);
            return;
        }
    }
    interrupt0();
}

public static boolean interrupted() {
    return currentThread().isInterrupted(true);
}
public boolean isInterrupted() {
    return isInterrupted(false);
}
private native boolean isInterrupted(boolean ClearInterrupted);

可以看到,interrupt方法通過設置中斷位來完成中斷。interrupted方法和isInterrupted方法都是通過調用native方法來檢測中斷的,interrupted是一個靜態(tài)方法,用來檢測當前線程是否被中斷,而且interrupted會清除該線程的中斷狀態(tài);isInterrupted是一個實例方法,可用來檢驗是否有線程被中斷,該方法不會改變中斷狀態(tài)。

一般來說,我們中斷線程的目的很可能是想停止線程執(zhí)行。怎么停止線程執(zhí)行呢?我們可以在判斷中斷置位后,用return退出run方法。但這樣設計并不優(yōu)雅,另外一種方式,就是拋出InterruptedException并在run方法里捕獲。捕獲后怎么處理也是件值得考慮的事,最好的方法是直接拋給調用者處理,但run方法是重寫方法,結構已固定,無法拋出異常,我們還可以在捕獲InterruptedException后重新中斷當前線程,讓調用者檢測。

五、線程相關方法 1.Object.wait()、Object.wait(long timeout)、Object.notify()、Object.notifyAll()

wait方法是掛起當前線程,釋放當前對象的控制權(釋放鎖),然后線程處于等待狀態(tài)。notify是通知正在等待對象控制權(鎖)的線程可以繼續(xù)運行。notifyAll是通知所有等待對象控制權的線程繼續(xù)運行。這幾個方法是基于monitor監(jiān)視器鎖來實現(xiàn)的,所以必須在同步塊內執(zhí)行。

2.Thread.sleep()、Thread.yield()、Thread.join()

sleep讓當前線程暫停指定時間。wait方法依賴于同步,sleep可以直接調用。因為sleep只是暫時讓出CPU的執(zhí)行權,并不釋放鎖,而wait需要釋放鎖。舉個簡單的例子:

public class WaitTest {
    private static Object o = new Object();
    
    static class Thread1 extends Thread{

        @Override
        public void run() {
            try {
                synchronized(o){
                    System.out.println("Thread1--start");
                    //o.wait();
                    Thread.sleep(2000);
                    System.out.println("Thread1--end");
                }
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
                System.out.println("a");
            }
        }
    }
    
    static class Thread2 extends Thread{

        @Override
        public void run() {
            synchronized(o){
                System.out.println("Thread2--start");
                o.notify();
                System.out.println("Thread2--end");
            }
        }
    }
    
    public static void main(String[] args) throws InterruptedException{
        Thread1 t1 = new Thread1();
        Thread2 t2 = new Thread2();
        t1.start();
        Thread.sleep(100);//保證t1先獲得鎖
        t2.start();
    }
}

在線程1里分別調用sleep和wait會有不同的結果,調用sleep時線程1不會釋放鎖,所以會打印完“Thread1 start”、“Thread1 end”,再進入線程2的打印。調用wait時,打印完“Thread1-start”,就會釋放鎖,這時線程2的打印得以繼續(xù)進行,會打印“Thread2 start”。

Thread.yield()方法會將當前線程從Running轉為Runnable,讓出當前對進程的使用,以便其他線程有機會執(zhí)行,不過調度器可以忽虐該方法,也不能指定暫停時間,一般只用來調試和測試。

Thread.join()方法用于將異步的線程“合并”為同步的線程,父線程等待子線程執(zhí)行完成后再執(zhí)行。其實并不算合并,而是調用join的線程進入限時等待,不斷檢查子線程狀態(tài),在子線程執(zhí)行完成后恢復執(zhí)行??匆幌滤膶崿F(xiàn)原理:

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;
        }
    }
}

可以看到,join是通過子線程不斷輪詢自己狀態(tài)一直到執(zhí)行完畢才返回繼續(xù)執(zhí)行父線程。

六、同步 1.synchronized、ReentrantLock與鎖優(yōu)化

在Java中,最基本的互斥同步手段就是synchronized關鍵字,synchronized自動提供一個鎖以及相關的條件。synchronized同步塊對于同一條線程來說是可重入的,其次,同步塊在已進入的線程執(zhí)行完之前,會阻塞后面其他線程的進入。前面提到,Java的線程是映射到系統(tǒng)原生線程上,阻塞或喚醒一個線程,都需要操作系統(tǒng)幫忙完成,需要從用戶態(tài)轉為核心態(tài),這需要耗費很長時間。因此synchronized是重量級操作,虛擬機本身會有一些優(yōu)化手段,比如在阻塞之前加入自旋等待過程,避免頻繁切入核心態(tài)之中。

重入鎖(ReentrantLock)與synchronized相似,具備一樣的線程重入性,一個表現(xiàn)為API層面的互斥鎖,另一個表現(xiàn)為原生語法層面的互斥鎖。ReetrantLock增加了一些功能:等待可中斷、公平鎖和鎖綁定多個條件。

公平鎖是指多個線程等待同一個鎖時,必須按照申請鎖的時間順序來依次獲得鎖,可以通過帶布爾值的構造函數(shù)要求使用公平鎖。

 /**
     * Creates an instance of {@code ReentrantLock} with the
     * given fairness policy.
     *
     * @param fair {@code true} if this lock should use a fair ordering policy
     */
public ReentrantLock(boolean fair) {
    sync = fair ? new FairSync() : new NonfairSync();
}

等待可中斷是指持有鎖的線程長期不釋放鎖的時候,正在等待的線程可以選擇放棄等待:

import java.util.concurrent.locks.ReentrantLock;

public class ReentrantLockTestOne {
    static int count = 0;
    static ReentrantLock lock = new ReentrantLock();
    
    public static void main(String[] args) throws Exception{
        Thread a = new Thread(new CountThread("a"));
        Thread b = new Thread(new CountThread("b"));
        a.start();
        Thread.sleep(100);//確保b線程后執(zhí)行,a能先獲得鎖
        b.start();
        Thread.sleep(500);//等待0.5s后,a線程還沒有釋放鎖,通過中斷放棄等待
        b.interrupt();
    }
    
    static class CountThread extends Thread{
        
        String name;

        CountThread(String name){
            this.name = name;
        }
                
        @Override
        public void run() {
            /*try{
                lock.lockInterruptibly();
            }catch(InterruptedException e){
                System.out.println("Thread "+name+" interrupted");
                return;
            }*/
            lock.lock();
            System.out.println(Thread.currentThread().isInterrupted());
            try{
                System.out.println("Thread "+name+" begin");
                for(int i=0; i<2000000; i++){
                    for(int j=0; j<100000; j++){
                        count++;
                    }
                }
                System.out.println("Thread "+name+" end");
            }finally{
                lock.unlock();
            }
        }
    }
}

我們先看lock.lock的執(zhí)行結果:

可以看見a線程執(zhí)行完后b才開始執(zhí)行,且b線程的中斷位已被置位。說明lock是阻塞式的獲取鎖,只有在成功獲取到鎖以后才處理中斷信息,并且怎么處理由調用端決定,lock只負責給中斷位置位。
再看一下lock.lockInterruptibly的執(zhí)行結果:

可以看到,lockInterruptibly會立即處理中斷信息,拋出InterruptedException,而不用等到獲取鎖。

鎖綁定多個條件是指一個ReentrantLock對象可以同時綁定多個Condition對象。在synchronized中,鎖對象的wait和notify其實實現(xiàn)一個隱含的條件,如果要和多個條件關聯(lián),必須額外添加鎖。

public class ReentrantLockTestTwo {
    
    static ReentrantLock lock = new ReentrantLock();
    
    static Condition productCondition = lock.newCondition();
    static Condition customerCondition = lock.newCondition();
    
    static Set set = new HashSet(8);
    
    public static void main(String[] args) {
        ProductThread pt = new ProductThread();
        CustomerThread ct = new CustomerThread();
        new Thread(pt).start();
        new Thread(pt).start();
        new Thread(ct).start();
        new Thread(ct).start();
    }
    
    static class ProductThread extends Thread{
        @Override
        public void run() {
            lock.lock();
            try{
                System.out.println("進入生產線程");
                for(;;){
                    Thread.sleep(1000);
                    if(set.size()>=6){
                        customerCondition.signalAll();
                        productCondition.await();
                    }else{
                        System.out.println("開始生產");
                        Object o = new Object();
                        set.add(o);
                        System.out.println("目前有"+set.size()+"個產品");
                    }
                }
            }catch(Exception e){
                
            }finally{
                lock.unlock();
            }
        }
    }
    
    static class CustomerThread extends Thread{
        @Override
        public void run() {
            lock.lock();
            try{
                System.out.println("進入使用線程");
                for(;;){
                    Thread.sleep(1000);
                    if(set.size()<=2){
                        productCondition.signalAll();
                        customerCondition.await();
                    }else{
                    System.out.println("開始使用");
                    Iterator it = set.iterator();
                    if(it.hasNext()){
                        Object o = it.next();
                        set.remove(o);
                    }
                    System.out.println("目前有"+set.size()+"個產品");
                    }
                    }
            }catch(Exception e){
            }finally{
                lock.unlock();
            }
        }
    }
}


上面展示了ReentrantLock鎖綁定多個條件??梢钥吹轿覀冊诋a品上加鎖并在鎖上新建了兩個條件:生產條件和使用條件。當產品數(shù)量多于6時,讓生產線程等待,小于2時,讓使用線程等待。從執(zhí)行結果可以看出,每次喚醒的線程只可能是生產或使用線程的一種,而并沒有喚醒這個鎖上的所有線程。

鎖優(yōu)化有幾種措施:自旋鎖與自適應鎖、鎖消除、鎖粗化、輕量級鎖和偏向鎖。

前面提到同步塊會阻塞其他線程,而線程的阻塞和恢復需要系統(tǒng)切換狀態(tài),耗費較長時間。所以如果持有鎖的線程很快就會釋放鎖時,我們并不需要讓等待線程阻塞,而是讓它執(zhí)行一個忙循環(huán),這就是所謂的自旋鎖。但自旋鎖雖然避免了線程切換的開銷,卻要占用處理器時間。當鎖被長時間占用時,自旋鎖除了浪費處理器資源就沒有作用了。JDK1.6引入了自適應的自旋鎖,有系統(tǒng)決定自旋時間,改善性能。

鎖消除是指虛擬機即時編譯器在運行時,對一些代碼上要求同步,但是被檢測到不可能存在共享數(shù)據(jù)競爭的鎖進行消除。鎖消除主要判定依據(jù)來源于逃逸分析的數(shù)據(jù)支持。

如果一系列的連續(xù)操作都對同一個對象反復加鎖和解鎖,甚至加鎖操作是出現(xiàn)在循環(huán)體中,頻繁地進行互斥同步操作也會導致不必要的性能損耗。這時可以鎖粗化。

偏向鎖是消除數(shù)據(jù)在無競爭情況下的同步,所謂偏向,是指其偏向第一個獲得它的線程。假設JVM啟用了偏向鎖,當鎖對象第一次被線程獲得的時候,虛擬機將會把對象頭的標志位設為“01”,即偏向模式。同時使用CAS操作把獲取到這個鎖的線程的ID記錄在對象的Mark Word中,如果CAS成功,持有偏向鎖的線程以后每次進入這個鎖相關的同步塊時,虛擬機都不再進行任何同步操作。第二個線程來訪問時,檢查原來持有對象鎖線程是否存活,若已介素則偏向鎖偏向第二個線程,否則第一個線程如果存活,通過線程棧檢查對象是否處于鎖定狀態(tài),如果無鎖,則撤銷偏向恢復到未鎖定對象,如果仍然鎖定,則升級為輕量級鎖。

輕量級鎖是在無競爭情況(個人認為是輕度競爭)下使用CAS操作去消除同步使用的互斥量,線程在執(zhí)行同步塊之前,虛擬機在當前線程的棧幀中建立Lock Record來存儲對象目前Mark Word的拷貝,然后JVM通過CAS替換對象Mark Word為Lock Record的指針。如果成功,對象處于輕量級鎖定,失敗說明存在額外線程競爭鎖,則嘗試自旋,如果自旋時間內還未獲得鎖,則開始膨脹,修改MarkWord為重量級鎖的指針,并且阻塞自己。

2.線程局部變量ThreadLocal

同步機制采用了“以時間換空間”的方式,而ThreadLocal采用了“以空間換時間”的方式。ThreadLocal會在每個線程中為變量創(chuàng)建一個副本,即每個線程內部都會有一個該變量,且在線程內部任何地方可以使用,線程之間互不影響,這樣需要在多線程使用的變量就不存在線程安全問題。

ThreadLocal本身并不存儲變量值,它本身其實只是一個鍵值對的鍵,用來讓線程從ThreadLocalMap中獲取Value,ThreadLocalMap是每個線程內部的容器。

可以看一下ThreadLocal的源碼:

public void set(T value) {
    Thread t = Thread.currentThread();
    ThreadLocalMap map = getMap(t);
    if (map != null)
        map.set(this, value);
    else
        createMap(t, value);
}
public T get() {
    Thread t = Thread.currentThread();
    ThreadLocalMap map = getMap(t);
    if (map != null) {
        ThreadLocalMap.Entry e = map.getEntry(this);
        if (e != null) {
            @SuppressWarnings("unchecked")
            T result = (T)e.value;
            return result;
        }
    }
    return setInitialValue();
}

同時,我們看一下ThreadLocalMap的源碼,會發(fā)現(xiàn)它的key使用的是ThreadLocal的弱引用。至于為什么用弱引用,是因為從上圖我們可以看見一共有兩條引用鏈到ThreadLocal變量,如果ThreadLocalRef置空,也就是程序不再訪問ThreadLocal變量了。此時如果key使用的是強引用,那么根據(jù)判斷對象存亡的可達性分析算法,ThreadLocal并不會被回收,因為還有一條GC root的引用鏈到ThreadLocal上;如果使用的是弱引用,我們知道弱引用只會存活到下一次JVM GC時,ThreadLocal就可以被回收。

static class Entry extends WeakReference> {
    /** The value associated with this ThreadLocal. */
    Object value;

    Entry(ThreadLocal k, Object v) {
        super(k);
        value = v;
    }
}

使用弱引用ThreadLocal固然可以被回收,但是帶來新的問題。ThreadLocal被回收后ThreadLocalMap中會出現(xiàn)key為null的Entry,意味著沒有辦法訪問這些key為null的Entry的value,如果當前線程遲遲不結束,value對應的對象不被回收,就會導致內存泄漏。從下面的代碼看到,ThreadLocal的set、get、remove方法在一些時機下會清理這些value,但這不及時,還是會有一些內存泄漏,最好的辦法時我們可以通過每次使用完ThreadLocal后,調用它的remove方法來避免這種情況。

public T get() {
    Thread t = Thread.currentThread();
    ThreadLocalMap map = getMap(t);
    if (map != null) {
        ThreadLocalMap.Entry e = map.getEntry(this);
        if (e != null) {
            @SuppressWarnings("unchecked")
            T result = (T)e.value;
            return result;
        }
    }
    return setInitialValue();
}

private Entry getEntry(ThreadLocal key) {
    int i = key.threadLocalHashCode & (table.length - 1);
    Entry e = table[i];
    if (e != null && e.get() == key)
        return e;
    else
        return getEntryAfterMiss(key, i, e);
}
private Entry getEntryAfterMiss(ThreadLocal key, int i, Entry e) {
    Entry[] tab = table;
    int len = tab.length;

    while (e != null) {
        ThreadLocal k = e.get();
        if (k == key)
            return e;
        if (k == null)
            expungeStaleEntry(i);
        else
            i = nextIndex(i, len);
        e = tab[i];
    }
    return null;
}
private int expungeStaleEntry(int staleSlot) {
    Entry[] tab = table;
    int len = tab.length;

    // expunge entry at staleSlot
    tab[staleSlot].value = null;
    tab[staleSlot] = null;
    size--;

    // Rehash until we encounter null
    Entry e;
    int i;
    for (i = nextIndex(staleSlot, len);
         (e = tab[i]) != null;
         i = nextIndex(i, len)) {
        ThreadLocal k = e.get();
        if (k == null) {
            e.value = null;
            tab[i] = null;
            size--;
        } else {
            int h = k.threadLocalHashCode & (len - 1);
            if (h != i) {
                tab[i] = null;

                // Unlike Knuth 6.4 Algorithm R, we must scan until
                // null because multiple entries could have been stale.
                while (tab[h] != null)
                    h = nextIndex(h, len);
                tab[h] = e;
            }
        }
    }
    return i;
}
3.Collections

Collections作為集合的工具類,除了提供一些有效的算法之外,還可以對集合進行包裝。其中一種就是非同步集合包裝成同步集合。

public static  Map synchronizedMap(Map m) {
    return new SynchronizedMap<>(m);
}
private static class SynchronizedMap
    implements Map, Serializable {
    private static final long serialVersionUID = 1978198479659022715L;

    private final Map m;     // Backing Map
    final Object      mutex;        // Object on which to synchronize

    SynchronizedMap(Map m) {
        this.m = Objects.requireNonNull(m);
        mutex = this;
    }

    SynchronizedMap(Map m, Object mutex) {
        this.m = m;
        this.mutex = mutex;
    }

    public int size() {
        synchronized (mutex) {return m.size();}
    }
    public boolean isEmpty() {
        synchronized (mutex) {return m.isEmpty();}
    }
    public boolean containsKey(Object key) {
        synchronized (mutex) {return m.containsKey(key);}
    }
    public boolean containsValue(Object value) {
        synchronized (mutex) {return m.containsValue(value);}
    }
    public V get(Object key) {
        synchronized (mutex) {return m.get(key);}
    }

    public V put(K key, V value) {
        synchronized (mutex) {return m.put(key, value);}
    }
    public V remove(Object key) {
        synchronized (mutex) {return m.remove(key);}
    }
    public void putAll(Map map) {
        synchronized (mutex) {m.putAll(map);}
    }
    public void clear() {
        synchronized (mutex) {m.clear();}
    }

    private transient Set keySet;
    private transient Set> entrySet;
    private transient Collection values;

    public Set keySet() {
        synchronized (mutex) {
            if (keySet==null)
                keySet = new SynchronizedSet<>(m.keySet(), mutex);
            return keySet;
        }
    }

    public Set> entrySet() {
        synchronized (mutex) {
            if (entrySet==null)
                entrySet = new SynchronizedSet<>(m.entrySet(), mutex);
            return entrySet;
        }
    }

    public Collection values() {
        synchronized (mutex) {
            if (values==null)
                values = new SynchronizedCollection<>(m.values(), mutex);
            return values;
        }
    }

    public boolean equals(Object o) {
        if (this == o)
            return true;
        synchronized (mutex) {return m.equals(o);}
    }
    public int hashCode() {
        synchronized (mutex) {return m.hashCode();}
    }
    public String toString() {
        synchronized (mutex) {return m.toString();}
    }

    // Override default methods in Map
    @Override
    public V getOrDefault(Object k, V defaultValue) {
        synchronized (mutex) {return m.getOrDefault(k, defaultValue);}
    }
    @Override
    public void forEach(BiConsumer action) {
        synchronized (mutex) {m.forEach(action);}
    }
    @Override
    public void replaceAll(BiFunction function) {
        synchronized (mutex) {m.replaceAll(function);}
    }
    @Override
    public V putIfAbsent(K key, V value) {
        synchronized (mutex) {return m.putIfAbsent(key, value);}
    }
    @Override
    public boolean remove(Object key, Object value) {
        synchronized (mutex) {return m.remove(key, value);}
    }
    @Override
    public boolean replace(K key, V oldValue, V newValue) {
        synchronized (mutex) {return m.replace(key, oldValue, newValue);}
    }
    @Override
    public V replace(K key, V value) {
        synchronized (mutex) {return m.replace(key, value);}
    }
    @Override
    public V computeIfAbsent(K key,
            Function mappingFunction) {
        synchronized (mutex) {return m.computeIfAbsent(key, mappingFunction);}
    }
    @Override
    public V computeIfPresent(K key,
            BiFunction remappingFunction) {
        synchronized (mutex) {return m.computeIfPresent(key, remappingFunction);}
    }
    @Override
    public V compute(K key,
            BiFunction remappingFunction) {
        synchronized (mutex) {return m.compute(key, remappingFunction);}
    }
    @Override
    public V merge(K key, V value,
            BiFunction remappingFunction) {
        synchronized (mutex) {return m.merge(key, value, remappingFunction);}
    }

    private void writeObject(ObjectOutputStream s) throws IOException {
        synchronized (mutex) {s.defaultWriteObject();}
    }
}

可以看見,其實包裝的原理很簡單,無非是對原來的所有操作加上同步鎖,這樣非同步集合就成了同步集合。

4.ReentrantReadWriteLock

讀寫鎖其實就是共享鎖和排它鎖。如果對資源加了寫鎖,其他線程無法再獲得讀鎖或寫鎖,但持有寫鎖的線程,可以對資源加讀鎖(鎖降級)。如果線程對資源加了讀鎖,其他線程可以繼續(xù)加讀鎖。舉個例子:幾個人一起開發(fā),SVN服務器上的代碼大家可以同時查看,但對同一段代碼的修改提交同時只能一個人操作。這里查看就需要讀鎖,提交就需要加寫鎖,如下代碼。

import java.util.concurrent.locks.ReentrantReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock.ReadLock;
import java.util.concurrent.locks.ReentrantReadWriteLock.WriteLock;

public class ReentrantReadWriteLockTest {
    
    static int readCount = 0;
    static int writeCount = 0;
    static ReentrantReadWriteLock lock = new ReentrantReadWriteLock();
    static String code = "hello world";
    static ReadLock readlock = lock.readLock();
    static WriteLock writelock = lock.writeLock();
    
    public static void main(String[] args) {
        ReadThread r = new ReadThread();
        WriteThread w = new WriteThread();
        for(int i=0; i<3; i++){
            new Thread(r).start();
            new Thread(w).start();
        }
    }
    
    static class ReadThread extends Thread{
        @Override
        public void run() {
            while(true){
                readlock.lock();
                try{
                    readCount ++;
                    System.out.println("同時有"+readCount+"個線程同時讀的內容: "+code);
                    String temp = new String(code);
                    try {
                        Thread.sleep(2000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println(code.equals(temp));
                    readCount --;
                }finally{
                    readlock.unlock();
                }
            }
        }
    } 
    
    static class WriteThread extends Thread{
        @Override
        public void run() {
            while(true){
                writelock.lock();
                try{
                    writeCount ++;
                    code = code + "a";
                    System.out.println("同時有"+writeCount+"個線程寫的內容: "+code);
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    writeCount --;
                }finally{
                    writelock.unlock();
                }
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    } 
}

我們來看一下運行結果:

可以看到,有多個線程同時讀取代碼,但任意時刻只有一個線程進行更改。且讀的時候不允許更改(代碼是通過比較前后兩次讀到的內容來驗證讀寫鎖不兼容的,這不夠嚴謹,暫時沒有想到更好例子)。至于鎖降級,因為在修改數(shù)據(jù)后寫線程沒有再用到數(shù)據(jù),所以上例中沒有用鎖降級,在此摘抄一段話來說明其必要性。

參考:《深入理解Java虛擬機》、《Java并發(fā)編程實戰(zhàn)》、《Java核心技術》。

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

轉載請注明本文地址:http://systransis.cn/yun/68754.html

相關文章

  • 線程編程完全指南

    摘要:在這個范圍廣大的并發(fā)技術領域當中多線程編程可以說是基礎和核心,大多數(shù)抽象并發(fā)問題的構思與解決都是基于多線程模型來進行的。一般來說,多線程程序會面臨三類問題正確性問題效率問題死鎖問題。 多線程編程或者說范圍更大的并發(fā)編程是一種非常復雜且容易出錯的編程方式,但是我們?yōu)槭裁催€要冒著風險艱辛地學習各種多線程編程技術、解決各種并發(fā)問題呢? 因為并發(fā)是整個分布式集群的基礎,通過分布式集群不僅可以大...

    mengera88 評論0 收藏0
  • 超詳細的Java面試題總結(二)之Java基礎知識

    摘要:超詳細的面試題總結一之基本知識多線程和虛擬機創(chuàng)建線程有幾種不同的方式你喜歡哪一種為什么繼承類實現(xiàn)接口應用程序可以使用框架來創(chuàng)建線程池實現(xiàn)接口。死亡線程方法執(zhí)行結束,或者因異常退出了方法,則該線程結束生命周期。死亡的線程不可再次復生。 超詳細的Java面試題總結(一)之Java基本知識 多線程和Java虛擬機 創(chuàng)建線程有幾種不同的方式?你喜歡哪一種?為什么? 繼承Thread類 實現(xiàn)R...

    wangjuntytl 評論0 收藏0
  • java 基礎 - 收藏集 - 掘金

    摘要:基礎知識復習后端掘金的作用表示靜態(tài)修飾符,使用修飾的變量,在中分配內存后一直存在,直到程序退出才釋放空間。將對象編碼為字節(jié)流稱之為序列化,反之將字節(jié)流重建成對象稱之為反序列化。 Java 學習過程|完整思維導圖 - 后端 - 掘金JVM 1. 內存模型( 內存分為幾部分? 堆溢出、棧溢出原因及實例?線上如何排查?) 2. 類加載機制 3. 垃圾回收 Java基礎 什么是接口?什么是抽象...

    makeFoxPlay 評論0 收藏0
  • JAVA 線程和并發(fā)基礎

    摘要:線程可以被稱為輕量級進程。一個守護線程是在后臺執(zhí)行并且不會阻止終止的線程。其他的線程狀態(tài)還有,和。上下文切換是多任務操作系統(tǒng)和多線程環(huán)境的基本特征。在的線程中并沒有可供任何對象使用的鎖和同步器。 原文:Java Multi-Threading and Concurrency Interview Questions with Answers 翻譯:并發(fā)編程網 - 鄭旭東 校對:方騰飛 多...

    vboy1010 評論0 收藏0
  • JAVA 線程和并發(fā)基礎面試問答

    摘要:多線程和并發(fā)問題是技術面試中面試官比較喜歡問的問題之一。線程可以被稱為輕量級進程。一個守護線程是在后臺執(zhí)行并且不會阻止終止的線程。其他的線程狀態(tài)還有,和。上下文切換是多任務操作系統(tǒng)和多線程環(huán)境的基本特征。 多線程和并發(fā)問題是 Java 技術面試中面試官比較喜歡問的問題之一。在這里,從面試的角度列出了大部分重要的問題,但是你仍然應該牢固的掌握Java多線程基礎知識來對應日后碰到的問題。(...

    dreamans 評論0 收藏0

發(fā)表評論

0條評論

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