+ ReentrantLock类的使用 
+ ReentrantReadWriteLock类的使用

1. 使用ReentrantLock类

ReentrantLock类能够实现线程之间同步互斥,并且在扩展功能上更加强大,例如嗅探技术多路分支通知等功能,在使用上比synchronized更加灵活。
ReentrantLock类具有完全互斥排他的效果,即同一时间只有一个线程在执行ReentrantLock.lock()方法后面的任务。

ReentrantLock实现了Lock接口,该接口包含以下方法:

public interface Lock {

    void lock();

    void lockInterruptibly() throws InterruptedException;

    boolean tryLock();

    boolean tryLock(long time, TimeUnit unit) throws InterruptedException;

    void unlock();

    Condition newCondition();
}

关键字 synchronized 与 wait() 和 notify() / notifyAll() 方法结合可以实现等待/通知模型, ReentrantLock 类也可以实现同样的功能,需要借助 Condition对象。

Condition 对象是JDK1.5中出现的技术,它有更好的灵活性,比如可实现多路通知功能,也就是在一个 Lock 对象里面可以创建多个 Condition (即对象监视器)实例,线程对象可以注册在指定的 Condition 中,从而可以有选择性地进行线程通知,在调度线程上更加灵活。

使用 notify() / notifyAll() 方法进行通知时,被通知的线程由JVM随机选择。通过 ReentrantLock 结合 Condition 可以实现选择性通知

synchronized 相当于整个 Lock 对象中只有一个单一的 Condition 对象,所有的线程都注册在它一个对象的身上。线程 notifyAll() 时,需要通知所有的 WAITING 线程,没有选择权。

Object 类中的 wait() 方法相当于 Condition 类中的 await() 方法。

Object 类中的 wait(long timeout) 方法相当于 Condition 类中的 await(long time,TimeUnit unit) 方法。

Object 类中的 notify() 方法相当于 Condition 类中的 signal() 方法。

Object 类中的 notifyAll() 方法相当于 Condition 类中的 signalAll() 方法。

公平锁与非公平锁

Lock 分为公平锁非公平锁公平锁表示线程获取锁的顺序是按照线程加锁的顺序来分配的,即先来先得的 FIFO先进先出顺序。非公平锁是一种获取锁的抢占机制,是随机获得锁的。

Lock方法

  1. 方法 int getHoldCount() 的作用是查询当前线程保持此锁定的个数,也就是调用 lock()方法的次数。
  2. 方法 int getQueueLength() 的作用是返回正在等待获取此锁定的线程估计数
  3. 方法 int getWaitQueueLength(Condition condition) 的作用是返回等待与此锁定相关的给定条件 Condition 的线程估计数。比如有5个线程,每个线程都执行了同一个 condition 对象的 await() 方法,则调用 getWaitQueueLength(condition) 方法返回的指为5.
  4. 方法 boolean hasQueuedThread(Thread thread) 的作用是查询指定的线程是否正在等待获取此锁定。
  5. 方法 boolean hasQueuedThreads() 的作用是差选是否有线程正在等待获取此锁定。
  6. 方法 boolean hasWaiters(Condition condition) 的作用是查询是否有线程正在等待与此锁定有关的 condition 条件。
  7. 方法 boolean isFair() 的作用是判断是不是公平锁。
  8. 方法 boolean isHeldByCurrentThread() 的作用是查询当前线程是否保持此锁定。
  9. 方法 boolean isLocked() 的作用是查询此锁定是否由任意线程保持。
  10. 方法 void lockInterruptibly() 的作用是:如果当前线程未被中断,则获取锁定;如果已经被中断则抛出异常( java.lang.InterruptedException ).
  11. 方法 boolean tryLock() 的作用是:仅在调用时锁未被另外线程保持的情况下,才获取此锁定。
  12. 方法 boolean tryLock(long timeout,TimeUnit unit) 的作用是:如果锁定在给定等待时间内没有被另一个线程保持,且当前线程未被中断,则获取该锁定。

使用Condition对象可以对线程执行的业务进行排序规划

2. 使用ReentrantReadWriteLock类

读写锁有两个锁:一个是读操作相关的锁,也称为共享锁;一个是写操作相关的锁,也称为排他锁

多个读锁之间不互斥;读锁与写锁互斥;写锁与写锁互斥。

package com.zxd.concurrent.learning;

import java.util.concurrent.locks.ReentrantReadWriteLock;

/**
* @author CoderZZ
* @Title: ${FILE_NAME}
* @Project: ConcurrentLearning
* @Package com.zxd.concurrent.learning
* @description: 读锁不互斥;读写、写写互斥
* @Version 1.0
* @create 2018-04-08 21:29
**/
public class ReentrantReadWriteLockTest { private ReentrantReadWriteLock reentrantReadWriteLock = new ReentrantReadWriteLock(); public void read(){
try {
reentrantReadWriteLock.readLock().lock();
System.out.println("获得读锁"+Thread.currentThread().getName()+" timestamp:"+System.currentTimeMillis());
Thread.sleep(10000);
} catch (InterruptedException e) {
e.printStackTrace();
}finally {
reentrantReadWriteLock.readLock().unlock();
}
} public void write(){
try{
reentrantReadWriteLock.writeLock().lock();
System.out.println("获得写锁"+Thread.currentThread().getName()+" timestamp:"+System.currentTimeMillis());
Thread.sleep(10000);
}catch (Exception e){
e.printStackTrace();
}finally {
reentrantReadWriteLock.writeLock().unlock();
}
} public static void main(String[] args){
ReentrantReadWriteLockTest reentrantReadWriteLockTest = new ReentrantReadWriteLockTest();
Runnable readRunnable = new Runnable() {
@Override
public void run() {
reentrantReadWriteLockTest.read();
}
};
Thread threadA = new Thread(readRunnable);
threadA.setName("A");
threadA.start();
Thread threadB = new Thread(readRunnable);
threadB.setName("B");
threadB.start();
Runnable writeRunnable = new Runnable() {
@Override
public void run() {
reentrantReadWriteLockTest.write();
}
};
Thread writeA = new Thread(writeRunnable);
writeA.setName("writeA");
writeA.start();
Thread writeB = new Thread(writeRunnable);
writeB.setName("writeB");
writeB.start();
}
}

运行结果如下:

Java Lock的使用的更多相关文章

  1. 转载:Java Lock机制解读

    Java Lock机制解读 欢迎转载: https://blog.csdn.net/chengyuqiang/article/details/79181229 1.synchronized synch ...

  2. Java Lock Example – ReentrantLock(java锁的例子)

    Welcome to Java Lock example tutorial. Usually when working with multi-threaded environment, we use ...

  3. Java Lock ReentrantLock ReentrantReadWriteLock

    Lock与Synchronized的区别:   1)Lock是一个接口,而synchronized是Java中的关键字,synchronized是内置的语言实现: 2)synchronized在发生异 ...

  4. Java Lock

    JVM中的另一种锁Lock的实现.与synchronized不同的是,Lock完全用Java写成,在java这个层面是无关JVM实现的.在java.util.concurrent.locks包中有很多 ...

  5. [Java] [Lock] [Synchronized VS ReentrantLock]

    Overview java编写多线程程序时,为了保证线程安全,需要对数据进行同步,经常用到的两种同步方式就是synchronized和重入锁ReentrantLock. 相似点 都是加锁方式 都是阻塞 ...

  6. Java Lock & Condition

    /* jdk1.5以后将同步和锁封装成了对象. 并将操作锁的隐式方式定义到了该对象中, 将隐式动作变成了显示动作. Lock接口: 出现替代了同步代码块或者同步函数.将同步的隐式锁操作变成现实锁操作. ...

  7. 深入分析 Java Lock 同步锁

    前言 Java 的锁实现,有 Synchronized 和 Lock.上一篇文章深入分析了 Synchronized 的实现原理:由Java 15废弃偏向锁,谈谈Java Synchronized 的 ...

  8. java lock锁住特定对象

    由于lock没有锁住特定对象的概念,该如何做到像synchronized同步块一样的效果呢? 答案:为每一个需要加锁的对像分配一把锁. 示例: List<User> users = new ...

  9. Java lock 能被中断, synchronized 不能被中断

    1.lock是可中断锁,而synchronized 不是可中断锁 线程A和B都要获取对象O的锁定,假设A获取了对象O锁,B将等待A释放对O的锁定, 如果使用 synchronized ,如果A不释放, ...

随机推荐

  1. tp5最强分页 自定义model,控制器引用。只显示一页

    1.不多逼逼 model 代码 <?phpnamespace app\common\model; use think\Model; class Fpage{ private $page;//当前 ...

  2. C# 防火墙操作之开启与关闭

    通过代码操作防火墙的方式有两种:一是代码操作修改注册表启用或关闭防火墙:二是直接操作防火墙对象来启用或关闭防火墙.不论哪一种方式,都需要使用管理员权限,所以操作前需要判断程序是否具有管理员权限. 1. ...

  3. Java调用Redis集群

    前文 需要使用以下jar包 Maven项目引用以下配置: <dependency> <groupId>org.apache.commons</groupId> &l ...

  4. 小米监控--Open-Falcon

    #安装下载软件yum -y install wget #更换aliyun源mv /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/CentOS-Ba ...

  5. flask 学习(二)

    安装了flask扩展 以及flask-bootstrap 默认情况下,flask在template文件夹寻找模板. flask 加载的是Jinja2模板,该模板引擎在flask中由函数render_t ...

  6. kubernetes排错系列:(二)、运行很久的kubernetes集群,创建出来的pod都是pending状态

    1.查看pod信息 # 查看pod 报错信息kubectl get pods发现pod的ip没有 生成,也没有分配到某个node节点 # 查看pod详细时间kubectl describe pods发 ...

  7. linux 基础 ls cd 目录含义

  8. 绝对好用Flash多文件大文件上传控件

    本实例采用的是Uploadify上传插件,.NET程序,源程序是从网上找的,但是有Bug,已经修改好,并标有部分注释.绝对好用,支持单文件.多文件上传,支持大文件上传,已经过多方面测试,保证好用. 以 ...

  9. 深入理解红黑树及C++实现

    介绍 红黑树是一种特殊的平衡二叉树(AVL),可以保证在最坏的情况下,基本动态集合操作的时间复杂度为O(logn).因此,被广泛应用于企业级的开发中. 红黑树的性质 在一棵红黑树中,其每个结点上增加了 ...

  10. logging.basicConfig函数各参数:

    import logging logging.basicConfig(level=logging.DEBUG, format='%(asctime)s %(filename)s[line:%(line ...