Lock锁的学习

本文介绍了Java中的Lock接口及其实现类ReentrantLock和ReadWriteLock,比较了Synchronized与Lock的区别,包括公平锁和非公平锁的概念,并通过示例展示了如何使用ReentrantLock创建公平锁的多线程同步场景。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

lock锁的定义:

是一个接口, 是控制多个线程对共享资源进行访问的工具。

Lock接口的实现类:

ReentrantLock(可重入锁)

ReadWriteLock(读写锁)

读锁:也称为共享锁,线程共享的,用于读 取操作的同步锁,锁对象是ReentrantReadWriteLock.ReadLock。 写锁:也称为独占锁,线程独占的,用于读取操作的同步锁,锁对象是ReentrantReadWriteLock.WriterLock。

Lock接口的实现:

// 1.创建锁对象
    private ReentrantLock lock = new ReentrantLock();


// 2.在同步代码块前调用锁定方法lock(),获得锁
            lock.lock();

try{

    // 同步代码块

}finally{
   // 3.在同步代码块执行完成后调用解锁方法unlock()释放锁
            lock.unlock();
}

注意:解锁操作为什么要放在try--finally结构中?

原因:我们获得锁之后,如果在同步代码块中发生了异常的话,就会导致解锁unlock()操作无法执行,这样可能会导致死锁,而放在try--finally结构中,就保证了解锁操作一定会执行,这样就避免了死锁问题。

Lock的基本方法:

 

方法详情:

void lockInterruptibly() throws InterruptedException

 

synchronized 与 Lock 的区别:

二者都是可重入锁,都可以实现线程之间的同步过程

1.synchronized是关键字,是JVM层面的,而Lock是一个接口,是JDK提供的API。

2.当一个线程获取了synchronized锁,其他线程便只能一直等待直至占有锁的线程释放锁。当发生以下情况之一线程才会释放锁:a.占有锁的线程执行完了该代码,然后释放对锁的占有。b.占有锁线程执行发生异常,此时JVM会让线程自动释放锁。c.占有锁线程进入 WAITING 状态从而释放锁,例如在该线程中调用wait()方法等 而 Lock锁只有执行unlock()方法才能释放锁

3、当多个线程读写文件时,读操作和写操作会发生冲突现象,写操作和写操作也会发生冲突现象,但是读操作和读操作不会发生冲突现象,但是如果采用synchronized进行同步的话,就会导致当多个线程都只是进行读操作时也只有获取锁的线程才能进行读操作,其他线程只能等待锁释放后才能读,Lock则可以实现当多个线程都只是进行读操作时,线程之间不会发生冲突,例如:ReentrantReadWriteLock()。

4、可以通过Lock得知线程有没有成功获取到锁

5、synchronized是可以手动在执行过程中,通过wait方法短暂释放资源,但是Lock不行,因为Lock在5底层没有对象的monitor。

公平锁和非公平锁:

公平锁:是指多个线程按照申请锁的顺序来获取锁,线程直接进入队列中排队,队列中的第一个线程才能获得锁。类似排队打饭,先来后到。

非公平锁:是指多个线程获取锁的顺序并不是按照申请锁的顺序,假设有一个线程此时加锁后正准备释放,这时候刚好又有一个线程进来获取锁,则有可能改线程就能获取该锁。(跟插队一样),如果没有刚好线程释放锁的话,则需要乖乖往后面排队,先来后到。就变成公平锁了。(synchronized和lock默认是非公平锁)

lock非公平锁设置:Lock lock=new ReentrantLock(false)。 默认是false,设置为true则为公平锁。

题目:创建三个线程,分别按顺序输入123456...

import java.util.concurrent.locks.ReentrantLock;

public class FirstThread extends Thread {
    ReentrantLock lock;

    public FirstThread(ReentrantLock lock){
        this.lock=lock;
    }
    @Override
    public void run() {
        while(Main.i<=10) {
            this.lock.lock();
            if(Main.i<=10) {
                System.out.println(this.getName()+" "+":"+Main.i);
                Main.i++;;
        }
            }
            this.lock.unlock();
    }

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

public class Main {
    public static int i=1;
    public static void main(String[] args) {
        //公平锁true
        ReentrantLock lock=new ReentrantLock(true);
        for(int i=0;i<3;i++) {
            FirstThread thread=new FirstThread(lock);
            thread.start();
        }
    }
}
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值