1. 程式人生 > 程式設計 >通過例項解析synchronized和lock區別

通過例項解析synchronized和lock區別

1,原始構成

synchronized是關鍵字,屬於JVM層面,通過wait,notify和notifyAll來排程執行緒。

Lock是具體類,是api層面的鎖。

2,使用方法

synchronized不需要使用者手動去釋放鎖, 當synchronized程式碼執行完後,系統會自動釋放鎖。

Lock需要使用者手動釋放鎖,否則會出現死鎖現象。需要lock和unlock配合try/finally語句塊來完成。

3,等待是否中斷

synchronized不可中斷,除非丟擲異常或者正常執行完畢。

Lock可中斷,可以設定超時方法或者呼叫中斷方法。

4,加鎖是否公平

synchronized非公平鎖。

Lock預設非公平鎖,可指定為公平鎖。

5,鎖繫結多個條件condition

synchronized沒有。

Lock用來分組喚醒需要喚醒的執行緒,可以精確喚醒,而不是像synchronized一樣要麼隨機喚醒一個執行緒,要麼全部喚醒。

Demo: 練習

  多執行緒之間按順序呼叫,實現A->B->C三個執行緒啟動,要求:AA列印5次,BB列印10次,CC列印15次,重複10遍。

package com.demo.lock;

import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

class ShareResource{
  int number = 1;
  private Lock lock = new ReentrantLock();
  
  private Condition c1 = lock.newCondition();
  private Condition c2 = lock.newCondition();
  private Condition c3 = lock.newCondition();
  
  public void print5(){
    lock.lock();
    try {
      while(number!=1){
        c1.await();
      }
      for(int i=1;i<=5;i++){
        System.out.println(Thread.currentThread().getName()+"\t"+i);
      }
      number = 2;
      c2.signal();
    } catch (Exception e) {
      e.printStackTrace();
    }finally{
      lock.unlock();
    }
  }
  
  public void print10(){
    lock.lock();
    try {
      while(number!=2){
        c2.await();
      }
      for(int i=1;i<=10;i++){
        System.out.println(Thread.currentThread().getName()+"\t"+i);
      }
      number = 3;
      c3.signal();
    } catch (Exception e) {
      e.printStackTrace();
    }finally{
      lock.unlock();
    }
  }
  
  public void print15(){
    lock.lock();
    try {
      while(number!=3){
        c3.await();
      }
      for(int i=1;i<=15;i++){
        System.out.println(Thread.currentThread().getName()+"\t"+i);
      }
      number = 1;
      c1.signal();
    } catch (Exception e) {
      e.printStackTrace();
    }finally{
      lock.unlock();
    }
  }
}
public class TestReentrantLock {

  public static void main(String[] args) {
    ShareResource shareResource = new ShareResource();
    
    new Thread(()->{
      for(int i=1;i<=10;i++){
        shareResource.print5();
      }
    },"AA").start();
    
    new Thread(()->{
      for(int i=1;i<=10;i++){
        shareResource.print10();
      }
    },"BB").start();
    
    new Thread(()->{
      for(int i=1;i<=10;i++){
        shareResource.print15();
      }
    },"CC").start();
  }
}

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援我們。