ホームページ >Java >&#&チュートリアル >Javaでリエントラントスピンロックを実装する方法

Javaでリエントラントスピンロックを実装する方法

WBOY
WBOY転載
2023-04-18 15:31:081469ブラウズ

説明

1. ロックを取得しようとするスレッドはブロックされず、ループを通じてロックを取得することを意味します。

2. 利点: コンテキスト切り替えの消費を削減します。

欠点: ループは CPU を消費します。

public class ReentrantSpinLock {
 
 
    private AtomicReference<Thread> owner = new AtomicReference<>();
 
    // 可重入次数
    private int count = 0;
 
    // 加锁
    public void lock() {
        Thread current = Thread.currentThread();
        if (owner.get() == current) {
            count++;
            return;
        }
        while (!owner.compareAndSet(null, current)) {
            System.out.println("--我在自旋--");
        }
    }
 
    //解锁
    public void unLock() {
        Thread current = Thread.currentThread();
        //只有持有锁的线程才能解锁
        if (owner.get() == current) {
            if (count > 0) {
                count--;
            } else {
                //此处无需CAS操作,因为没有竞争,因为只有线程持有者才能解锁
                owner.set(null);
            }
        }
    }
 
    public static void main(String[] args) {
        ReentrantSpinLock spinLock = new ReentrantSpinLock();
        Runnable runnable = () -> {
            System.out.println(Thread.currentThread().getName() + "开始尝试获取自旋锁");
            spinLock.lock();
            try {
                System.out.println(Thread.currentThread().getName() + "获取到了自旋锁");
                Thread.sleep(4000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            } finally {
                spinLock.unLock();
                System.out.println(Thread.currentThread().getName() + "释放了了自旋锁");
            }
        };
        Thread thread1 = new Thread(runnable);
        Thread thread2 = new Thread(runnable);
        thread1.start();
        thread2.start();
    }
}

以上がJavaでリエントラントスピンロックを実装する方法の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事はyisu.comで複製されています。侵害がある場合は、admin@php.cn までご連絡ください。