JAVA 锁

锁的概念

Java中的锁是控制资源访问的一种方式。它弥补了synchronized的可操作性不强的不足。

Java的锁都实现了Lock接口。Lock结构定义了锁的基本操作。

函数 解释
void lock() 获取锁,如果锁被其他线程占用,则等待
void lockInterruptibly() throws InterruptedException 获取锁,如果锁被其他线程占用,则等待。当该线程处于等待状态的时候,可以被interrupt()中断
boolean tryLock() 尝试获取锁,如果成功,返回true,如果失败,返回false
boolean tryLock(long time, TimeUnit unit) throws InterruptedException; 与tryLock的不同是,使用该方法,如果无法获取锁,会等待一段时间,在这段时间内,如果无法获取,则返回false。
Condition newCondition() 与wait,notify 类似,必须和锁一起配合使用

lock的基本用法:

  Lock lock = new ***();
lock.lock();
try {
//业务逻辑
} finally {
lock.unlock();
}

tryLock 的基本用法:

Lock lock = ***;
if(lock.tryLock()) {
try{
//处理业务
}catch(Exception ex){ }finally{
lock.unlock();
}
}else {
//获取不到锁,做其他业务
}

lockInterruptibly 的基本用法:

https://www.zhihu.com/question/36771163

package com.thread;

import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; public class TestLockInterruptibly
{ // @Test
public void test3() throws Exception
{
final Lock lock = new ReentrantLock();
lock.lock(); Thread t1 = new Thread(new Runnable()
{
@Override
public void run()
{
try
{
lock.lockInterruptibly();
}
catch(InterruptedException e)
{
System.out.println(Thread.currentThread().getName() + " interrupted.");
}
}
}, "child thread -1"); t1.start();
Thread.sleep(1000); t1.interrupt(); Thread.sleep(1100);
} public static void main(String[] args) throws Exception
{
new TestLockInterruptibly().test3();
}
}

上面是Lock接口的基本介绍,在实际使用中,主要有一下几种锁:

  • 重入锁
  • 读写锁

这些锁实现了Lock接口,为用户提供了统一的调用方法。内部实现基本都是通过聚合了一个队列同步器AbstractQueuedSynchronizer的子类来完成线程访问控制的。

ReentrantLock

ReentrantLock是一个重入锁,它表示该锁能够支持一个线程对资源的重复加锁。ReentrantLock还支持公平和非公平。

package com.thread;

import java.util.*;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; public class FairAndUnfairTest { private static Lock fairLock = new ReentrantLock2(true);
private static Lock unfairLock = new ReentrantLock2(false); public void fair() {
testLock(fairLock);
} public void unfair() {
testLock(unfairLock);
} public void testLock(Lock lock) {
for (int i = 0 ; i < 5; i++) {
Job job = new Job(lock);
job.start();
} } private static class Job extends Thread {
private Lock lock;
public Job(Lock lock) {
this.lock = lock;
}
public void run() {
lock.lock();
try {
System.out.print("current thread name is: " + this.getName());
print_list(lock);
} finally {
lock.unlock();
}
lock.lock();
try {
System.out.print("current thread name is: " + this.getName());
print_list(lock);
} finally {
lock.unlock();
} } private void print_list(Lock lock) {
ReentrantLock2 t_lock = (ReentrantLock2)lock;
Collection<Thread> threads = t_lock.getQueuedThreads();
System.out.print(" the thread in queue is : ");
for (Thread t : threads) {
System.out.print(t.getName() + ' ');
}
System.out.print('\n');
} } private static class ReentrantLock2 extends ReentrantLock { public ReentrantLock2(boolean fair) {
super(fair);
} public Collection<Thread> getQueuedThreads() {
List<Thread> arrayList = new ArrayList<Thread>(super.getQueuedThreads()); Collections.reverse(arrayList);
return arrayList;
}
} public static void main(String[] args) {
FairAndUnfairTest test = new FairAndUnfairTest();
test.unfair(); }
}

ReentrantLock通过在构造函数中传入true和false设置为公平或者非公平。如果是公平锁,就会每次从等待队列中获取最前的线程使用锁,否则就当前线程很可能再次获得锁。

公平锁有助于平均分配资源,但是线程不断切换会造成更多的开销。

ReentrantReadWriteLock

读写锁在同一个时刻允许多个读线程进行访问,但是在写线程访问的时候,所有的读线程和其它写线程均被阻塞。

 package com.thread;

import java.util.HashMap;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantReadWriteLock; public class Cache {
static Map<String, Object> map = new HashMap<String, Object>(); static ReentrantReadWriteLock rwl = new ReentrantReadWriteLock(); static Lock r = rwl.readLock();
static Lock w = rwl.writeLock(); public static final Object get(String key) {
r.lock();
try {
return map.get(key);
} finally {
r.unlock();
}
} public static final Object put (String key, Objects value) {
w.lock();
try {
return map.put(key, value);
} finally {
w.unlock();
}
} public static final void clear() {
w.lock();
try {
map.clear();
} finally {
w.unlock();
}
}
}

上面的这段代码就是使用了读写锁来保证map的线程安全。

参考:

1.JAVA并发的艺术
2.http://www.cnblogs.com/dolphin0520/p/3923167.html

JAVA 锁的更多相关文章

  1. java 锁!

    问题:如何实现死锁. 关键: 1 两个线程ta.tb 2 两个对象a.b 3 ta拥有a的锁,同时在这个锁定的过程中,需要b的锁:tb拥有b的锁,同时在这个锁定的过程中,需要a的锁: 关键的实现难点是 ...

  2. Java锁(一)之内存模型

    想要了解Java锁机制.引发的线程安全问题以及数据一致性问题,有必要了解内存模型,机理机制了解清楚了,这些问题也就应声而解了. 一.主内存和工作内存 Java内存模型分为主内存和工作内存,所有的变量都 ...

  3. Java锁的种类

    转载自:---->http://ifeve.com/java_lock_see/ Java锁的种类以及辨析锁作为并发共享数据,保证一致性的工具,在JAVA平台有多种实现(如 synchroniz ...

  4. JAVA锁的可重入性

    机制:每个锁都关联一个请求计数器和一个占有他的线程,当请求计数器为0时,这个锁可以被认为是unhled的,当一个线程请求一个unheld的锁时,JVM记录锁的拥有者,并把锁的请求计数加1,如果同一个线 ...

  5. JAVA 锁之 Synchronied

    ■ Java 锁 1. 锁的内存语义 锁可以让临界区互斥执行,还可以让释放锁的线程向同一个锁的线程发送消息 锁的释放要遵循 Happens-before 原则(锁规则:解锁必然发生在随后的加锁之前) ...

  6. java锁与监视器概念 为什么wait、notify、notifyAll定义在Object中 多线程中篇(九)

    在Java中,与线程通信相关的几个方法,是定义在Object中的,大家都知道Object是Java中所有类的超类 在Java中,所有的类都是Object,借助于一个统一的形式Object,显然在有些处 ...

  7. 自己动手写java锁

    1.LockSupport的park和unpark方法的基本使用,以及对线程中断的响应性 LockSupport是JDK中比较底层的类,用来创建锁和其他同步工具类的基本线程阻塞原语.java锁和同步器 ...

  8. Java 锁的学习

    个人学习整理,所有资料均来源于网络,非原创. 死锁的四个必要条件:互斥条件(Mutual exclusion):资源不能被共享,只能由一个进程使用.请求与保持条件(Hold and wait):已经得 ...

  9. java锁——wait,notify,synchronized

    背景:这篇博客用来总结java锁相关的知识点,平时还是要自己多加练习 wait 和 notify以及notifyAll (1).方法介绍1.wait.notify以及notifyAll都是Object ...

随机推荐

  1. UITableview刷新某一个cell或section

    //一个section刷新 NSIndexSet *indexSet=[[NSIndexSet alloc]initWithIndex:2]; [tableview reloadSections:in ...

  2. Minifilter微过滤框架:框架介绍以及驱动层和应用层的通讯

    minifilter是sfilter后微软推出的过滤驱动框架.相比于sfilter,他更容易使用,需要程序员做的编码更简洁. 系统为minifilter专门制作了一个过滤管理器,这个管理器本身其实是一 ...

  3. Linear Regression

    大学时候学物理实验的时候接触过线性回归,现在忘记了...还得重新拾起来.学习不扎实耽误了多少时光... sigh Suppose that you time a program as a functi ...

  4. Hadoop HDFS文件系统通过java FileSystem 实现上传下载等

    package linlintest; import java.io.File; import java.io.FileOutputStream; import java.io.IOException ...

  5. 编辑器Emacs下载网址(中国镜像)

      Root gnu emacs windows File Name ↓ File Size ↓ Date ↓ Parent directory/ - - README 14K 2014-Nov-15 ...

  6. SPOJ 227 Ordering the Soldiers 线段树 / 树状数组

    题意:设原数组为a[i],pos[i]代表第 i 个位置之前有多少个数比a[i]大,求原数组a[i]. 这个题意是看了别人的题解才明白,我自己没读出来…… 方法:假设我们从左往右放,因为后面的数还有可 ...

  7. 解决 ko mapping 数组无法添加新对象的问题

    这两天页面模板化的进程有些放缓,使用 ko mapping 插件的情形多了起来.组员经常问到的问题即是往 ko mapping 数组添加新对象时,报找不到方法的错误:而使用 ko.observable ...

  8. Docker+K8S实践

    一.运维角度: (一)镜像: 1. 避免依赖过深.不要在基础镜像上加太多产生其他的镜像,我觉得这块最多是三四层. 一层是base景像再往上是工具.中间件这样的,再往上一层就是你自己的程序,再多就比较乱 ...

  9. 如何通过session控制单点登录

    web服务器为每一个浏览器实例对应一个session.这个session有自己的一个独立id,这个id保存在浏览器的cookie中(这个cookie貌似随着这个浏览器实例的关闭而清除),访问web服务 ...

  10. J2EE开发之常用开源项目介绍

    主要就我所了解的J2EE开发的框架或开源项目做个介绍,可以根据需求选用适当的开源组件进行开发.主要还是以Spring为核心,也总结了一些以前web开发常用的开源工具和开源类库 1持久层: 1)Hibe ...