Java多线程同步控制

关键字Synchronized

关键字SynchronizedObject.wat()Object.notify()是在jdk1.5之前用的多线程同步控制的方式,jdk1.5之后就提供了如下的java.util.concurrent工具包(简称为juc并发工具包),可以利用如下的新的工具来实现多线程间的同步。

如下代码,可以知道其使用场景及作用:

// 类T
class T {
    // 静态对象,是在类上的,只有一个
    private static Object lock = new Object();

    @SneakyThrows
    public void doSomething(){
        // 锁静态对象,只有一个,是系统一个锁
        synchronized (lock){
            lock.wait();
        }
    }
    public void doSomething2(){
        // 锁类对象,只有一个,是同一个锁
        synchronized (T.class){
           // 做一些事情
            lock.notify();
        }
    }
    // 修饰在实例方法上,实则是在调用对象上加锁,调用对象不同则锁不同
    public synchronized void doSomething3(){

    }

    // 修饰在静态方法上,实则是在当前类对象上加锁,只有一个锁
    public static synchronized void doSomething4(){

    }
}

重入锁 ReentrantLock

定义

重入锁可以允许一个线程连续2次获取同一把锁,当然解锁也需要解锁2次。(关键字synchronized也是重入锁)

重要方法

  • lock() 获取锁,如果锁被占用,则等待
  • lockInterruptibly() 获取锁,但优先响应中断
    可以利用中断来直接退出等待锁,释放资源,防止死锁
  • tryLock() 尝试获取锁,理解返回结果;获取成功返回true,失败返回false
  • tryLock(timeout,unit) 给定时间内尝试获取锁
  • unlock() 释放锁

公平锁

构造方法内传递fair=true,则返回的是公平锁,默认是非公平锁,关键字synchronized也是非公平锁。

ReentrantLock lock = new ReentrantLock(true);

公平锁会按锁的线程等待队列的顺序,公平的让线程获取到锁,所以性能会比非公平锁低。

等待队列 Condition

Condition是重入锁的好搭档,让锁可以支持多个等待队列。提供了和Object.wait()Object.notify()和对应功能的方法Condition.await()Condition.signal()Condition.signalAll()

具体使用可参考ArrayBlockQueue的实现。

信号量 Semaphore

无论是内部锁synchronized还是重入锁ReentrantLock,一次都只允许一个线程访问临界区资源。而
Semaphore可以同时允许多个线程同时访问某一个资源。
构造方法如下:

public Semaphore(int permits);// 第一个参数信号量数量
public Semaphore(int permits, boolean fair); // 第二个参数是否公平

主要方法如下:

public void acquire(); // 获取许可,阻塞直到获取到或者线程被中断
public void acquireUninterruptibly(); // 获取许可,阻塞直到获取到,不响应中断
public boolean tryAcquire(); // 尝试获取许可,直接返回结果
public boolean tryAcquire(long timeout,TimeUnit unit); // 在限定时间内尝试获取许可
public void release(); // 释放许可

是一个有效的流量控制工具,它基于AQS共享锁实现。常常用它来控制对有限资源的访问。

  • 每次使用资源前,先申请一个信号量,如果资源数不够,就会阻塞等待;
  • 每次释放资源后,就释放一个信号量。

读写锁 ReadWriteLock

读写分离锁可以减少锁竞争,提升性能,适合在读多写少的场景使用。
是JDK5中提供的,实现类有ReentrantReadWriteLock

ReadWriteLock readWriteLock = new ReentrantReadWriteLock();
Lock readLock = readWriteLock.readLock();// 读锁,共享锁
Lock writeLock = readWriteLock.writeLock();// 写锁,独占锁

倒计数器 CountDownLatch

CountDownLatch是一个非常实用的多线程控制工具类,可以让某一个线程等待,直到倒计数器结束。典型的场景,如火箭发射时的倒数计时(等待各个发射细节就位)。

CountDownLatch countDownLatch = new CountDownLatch(3);
// 注意:相关检查是多线程并发执行的
// 检查1
// doCheck1()
countDownLatch.countDown();
// 检查2
// doCheck2()
countDownLatch.countDown();
// 检查3
// doCheck3()
countDownLatch.countDown();
// 等待检查完成,再执行发射逻辑
countDownLatch.await();
// 点火发射
// doFire()

循环栅栏 CyclicBarrier

CyclicBarrier也是一种多线程并发控制的工具,和CountDownLatch类似,前者可以重复使用,而后者只使用一次。

例子:

public static void main(String[] args) {

    CyclicBarrier cyclicBarrier1 = new CyclicBarrier(2, () -> {
        System.out.println("满2个,group1执行完了");
    });
    CyclicBarrier cyclicBarrier2 = new CyclicBarrier(3, () -> {
        System.out.println("满3个,group2执行完了");
    });
    for(int i=0;i<100;i++) {
        boolean b = i % 2 == 0;
        Thread t1 = new Thread(() -> {
            try {
                if(b){
                    // 偶数,则到篱笆1
                    cyclicBarrier1.await();
                }else{
                    // 偶数,则到篱笆2
                    cyclicBarrier2.await();
                }

            } catch (InterruptedException e) {
                e.printStackTrace();
            } catch (BrokenBarrierException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName() + "跑完了");

        });
        t1.setName("t"+i);
        t1.start();
    }
}

输出:

满2个,group1执行完了
t2跑完了
t0跑完了
满3个,group2执行完了
t5跑完了
t1跑完了
t3跑完了
满2个,group1执行完了
t6跑完了
t4跑完了
满2个,group1执行完了
t10跑完了
t8跑完了

阻塞工具类 LockSupport

LockSupport是一个非常方便使用的线程阻塞工具,可以在任意线程内让线程阻塞,不需要先获得某个对象的锁,也不会抛出中断异常。(底层实现类似信号量_counter计数,当park时,这个变量置为了0,当unpark时,这个变量置为1)

LockSupport.park(); // 阻塞当前线程,也有限时等待,还支持中断响应
LockSupport.unpark(Thread.currentThread()); // 恢复某个线程

参考文档

  • 书籍:葛一鸣 *《Java高并发程序设计第二版》
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值