Java Concurrency - ล็อคอินเทอร์เฟซ

อินเตอร์เฟส java.util.concurrent.locks.Lock ถูกใช้เพื่อเป็นกลไกการซิงโครไนซ์เธรดที่คล้ายกับบล็อกที่ซิงโครไนซ์ กลไกการล็อคแบบใหม่มีความยืดหยุ่นมากกว่าและมีตัวเลือกมากกว่าบล็อกที่ซิงโครไนซ์ ความแตกต่างหลักระหว่างล็อคและบล็อกซิงโครไนซ์มีดังต่อไปนี้ -

  • Guarantee of sequence- บล็อกที่ซิงโครไนซ์ไม่ได้รับประกันลำดับใด ๆ ที่เธรดรอจะได้รับการเข้าถึง ล็อคอินเทอร์เฟซจัดการมัน

  • No timeout- บล็อกที่ซิงโครไนซ์ไม่มีตัวเลือกในการหมดเวลาหากไม่ได้รับการล็อก ล็อคอินเทอร์เฟซมีตัวเลือกดังกล่าว

  • Single method - บล็อกที่ซิงโครไนซ์จะต้องมีอยู่อย่างครบถ้วนภายในวิธีการเดียวในขณะที่สามารถเรียกวิธีการล็อก () และปลดล็อก () ของอินเทอร์เฟซการล็อกด้วยวิธีการต่างๆ

วิธีการล็อค

ต่อไปนี้เป็นรายการวิธีการสำคัญที่มีอยู่ในคลาส Lock

ซีเนียร์ วิธีการและคำอธิบาย
1

public void lock()

ได้รับการล็อค

2

public void lockInterruptibly()

ได้รับการล็อกเว้นแต่เธรดปัจจุบันจะถูกขัดจังหวะ

3

public Condition newCondition()

ส่งคืนอินสแตนซ์ Condition ใหม่ที่เชื่อมโยงกับอินสแตนซ์ Lock นี้

4

public boolean tryLock()

จะได้รับการล็อคก็ต่อเมื่อมันว่างในเวลาที่เรียกใช้

5

public boolean tryLock()

จะได้รับการล็อคก็ต่อเมื่อมันว่างในเวลาที่เรียกใช้

6

public boolean tryLock(long time, TimeUnit unit)

ได้รับการล็อกหากว่างภายในเวลารอที่กำหนดและเธรดปัจจุบันไม่ถูกขัดจังหวะ

7

public void unlock()

ปลดล็อค

ตัวอย่าง

โปรแกรม TestThread ต่อไปนี้แสดงวิธีการเหล่านี้บางส่วนของอินเทอร์เฟซ Lock ที่นี่เราได้ใช้ lock () เพื่อรับล็อคและปลดล็อก () เพื่อคลายล็อก

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

class PrintDemo {
   private final Lock queueLock = new ReentrantLock();

   public void print() {
      queueLock.lock();

      try {
         Long duration = (long) (Math.random() * 10000);
         System.out.println(Thread.currentThread().getName() 
            + "  Time Taken " + (duration / 1000) + " seconds.");
         Thread.sleep(duration);
      } catch (InterruptedException e) {
         e.printStackTrace();
      } finally {
         System.out.printf(
            "%s printed the document successfully.\n", Thread.currentThread().getName());
         queueLock.unlock();
      }
   }
}

class ThreadDemo extends Thread {
   PrintDemo  printDemo;

   ThreadDemo(String name,  PrintDemo printDemo) {
      super(name);
      this.printDemo = printDemo;
   }   

   @Override
   public void run() {
      System.out.printf(
         "%s starts printing a document\n", Thread.currentThread().getName());
      printDemo.print();
   }
}

public class TestThread {

   public static void main(String args[]) {
      PrintDemo PD = new PrintDemo();

      ThreadDemo t1 = new ThreadDemo("Thread - 1 ", PD);
      ThreadDemo t2 = new ThreadDemo("Thread - 2 ", PD);
      ThreadDemo t3 = new ThreadDemo("Thread - 3 ", PD);
      ThreadDemo t4 = new ThreadDemo("Thread - 4 ", PD);

      t1.start();
      t2.start();
      t3.start();
      t4.start();
   }
}

สิ่งนี้จะให้ผลลัพธ์ดังต่อไปนี้

เอาต์พุต

Thread - 1  starts printing a document
Thread - 4  starts printing a document
Thread - 3  starts printing a document
Thread - 2  starts printing a document
Thread - 1   Time Taken 4 seconds.
Thread - 1  printed the document successfully.
Thread - 4   Time Taken 3 seconds.
Thread - 4  printed the document successfully.
Thread - 3   Time Taken 5 seconds.
Thread - 3  printed the document successfully.
Thread - 2   Time Taken 4 seconds.
Thread - 2  printed the document successfully.

เราใช้คลาส ReentrantLock เป็นการใช้อินเทอร์เฟซ Lock ที่นี่ คลาส ReentrantLock อนุญาตให้เธรดล็อกเมธอดแม้ว่าจะมีการล็อกเมธอดอื่นอยู่แล้วก็ตาม