JAVA基础知识|synchronized和lock
一、synchronized
是jvm的一个关键字,使用过程均由jvm控制
有三种使用方式:
- 修饰实例方法,作用于当前实例加锁,进入同步代码前要获得当前实例的锁
- 修饰代码块,同方法
- 修饰静态方法,作用于当前类加锁,进入同步代码前要获得当前类对象的锁
1.1、实例方法
作用于实例,会阻塞其他线程访问本实例,但不会影响其他线程访问其他实例
如果实例中有多个synchronized修饰的方法,当其中一个方法被访问时,其他方法也不可以被访问,所以我们在用synchronized关键字的时候,能缩小代码段的范围就尽量缩小,能在代码段上加同步就不要再整个方法上加同步
【示例】
import java.util.concurrent.TimeUnit; /**
* Created by jyy on 2018/6/21.
*/
public class SyncTest { public synchronized void test1(){
System.out.println("test1");
try {
TimeUnit.SECONDS.sleep(10);
}catch(InterruptedException ie){
System.out.println("中断异常");
}
System.out.println("test1");
} public synchronized void test2(){
System.out.println("test2");
} public void test3(){
System.out.println("test3");
}
}
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; /**
* Created by jyy on 2018/4/19.
*/
public class Test { public static void main(String[] args){
//需要声明为final类型
final SyncTest syncTest = new SyncTest();
//线程池
ExecutorService executorService = Executors.newCachedThreadPool();
executorService.execute(new Runnable() {
@Override
public void run() {
syncTest.test1();
}
});
executorService.execute(new Runnable() {
@Override
public void run() {
syncTest.test2();
}
});
executorService.execute(new Runnable() {
@Override
public void run() {
syncTest.test3();
}
});
}
}
执行结果:
test1
test3
test1
test2
1.2、代码块
作用于实例,同上
【示例】
//新增三个方法
public void test4(){
System.out.println("test4");
synchronized(this){
try {
TimeUnit.SECONDS.sleep(10);
}catch(InterruptedException ie){
System.out.println("中断异常");
}
System.out.println("test4");
}
} public void test5(){
System.out.println("test5");
synchronized(this){
System.out.println("test5");
}
} public void test6(){
System.out.println("test6");
}
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; /**
* Created by jyy on 2018/4/19.
*/
public class Test { public static void main(String[] args){
//需要声明为final类型
final SyncTest syncTest = new SyncTest();
//线程池
ExecutorService executorService = Executors.newCachedThreadPool();
executorService.execute(new Runnable() {
@Override
public void run() {
syncTest.test4();
}
});
executorService.execute(new Runnable() {
@Override
public void run() {
syncTest.test5();
}
});
executorService.execute(new Runnable() {
@Override
public void run() {
syncTest.test6();
}
});
}
}
执行结果:
test4
test5
test6
test4
test5
从执行结果上,可以看出,当访问关键字synchronized修饰的代码块时,其他synchronized修饰的代码块也会被锁住
1.3、静态方法
作用于类,会阻塞其他线程访问本类的同步方法,即使是不同的实例也不行
【示例】
//新增方法
public synchronized static void test7(){
int random= (new Random()).nextInt(100);
System.out.println(random);
try {
TimeUnit.SECONDS.sleep(10);
}catch(InterruptedException ie){
System.out.println("中断异常");
}
System.out.println(random);
}
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; /**
* Created by jyy on 2018/4/19.
*/
public class Test { public static void main(String[] args){
//线程池
ExecutorService executorService = Executors.newCachedThreadPool();
executorService.execute(new Runnable() {
@Override
public void run() {
SyncTest.test7();
}
});
executorService.execute(new Runnable() {
@Override
public void run() {
SyncTest.test7();
}
});
}
}
执行结果:
69
69
67
67
二、lock
Lock是一个接口,其中lock()、lockInterruptibly()、tryLock()、tryLock(long time, TimeUnit unit)方法可以获取锁,unlock()用来释放锁
如果采用Lock,必须主动去释放锁,并且在发生异常时,不会自动释放锁。因此一般来说,使用Lock必须在try{}catch{}块中进行,并且将释放锁的操作放在finally块中进行,以保证锁一定被被释放,防止死锁的发生。
2.1、ReentrantLock
ReentrantLock是实现了Lock接口的类,通过它,我们来看下四种获取锁方法的区别
2.1.1、lock()
【示例1】
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; /**
* Created by jyy on 2018/6/25.
*/
public class ReentrantLockTest {
private Lock lock = new ReentrantLock();//需要声明为全局变量 public void test1(){
lock.lock();//获取锁
try {
System.out.println(Thread.currentThread()+"得到了锁");
TimeUnit.SECONDS.sleep(5);
} catch (Exception e) {
System.out.println("发生异常");
}finally {
System.out.println(Thread.currentThread()+"释放了锁");
lock.unlock();
}
}
}
//线程池
ExecutorService executorService = Executors.newCachedThreadPool();
final ReentrantLockTest reentrantLockTest = new ReentrantLockTest();
executorService.execute(new Runnable() {
@Override
public void run() {
reentrantLockTest.test1();
}
});
executorService.execute(new Runnable() {
@Override
public void run() {
reentrantLockTest.test1();
}
});
执行结果:
Thread[pool-1-thread-1,5,main]得到了锁
Thread[pool-1-thread-1,5,main]释放了锁
Thread[pool-1-thread-2,5,main]得到了锁
Thread[pool-1-thread-2,5,main]释放了锁
【示例2】
//线程池
ExecutorService executorService = Executors.newCachedThreadPool();
final ReentrantLockTest reentrantLockTest = new ReentrantLockTest();
executorService.execute(new Runnable() {
@Override
public void run() {
reentrantLockTest.test1();
}
});
final ReentrantLockTest reentrantLockTest1 = new ReentrantLockTest();
executorService.execute(new Runnable() {
@Override
public void run() {
reentrantLockTest1.test1();
}
});
执行结果:
Thread[pool-1-thread-1,5,main]得到了锁
Thread[pool-1-thread-2,5,main]得到了锁
Thread[pool-1-thread-1,5,main]释放了锁
Thread[pool-1-thread-2,5,main]释放了锁
示例2说明Lock锁住的是实例,而不是类
2.1.2、tryLock()
tryLock()尝试获取锁,获取不到返回false,不影响代码继续执行
【示例】
//新增方法
public void test2(){
if(lock.tryLock())//获取锁
{
try {
System.out.println(Thread.currentThread() + "得到了锁");
TimeUnit.SECONDS.sleep(5);
} catch (Exception e) {
System.out.println("发生异常");
} finally {
System.out.println(Thread.currentThread() + "释放了锁");
lock.unlock();
}
}else{
System.out.println(Thread.currentThread() + "获取锁失败");
}
}
//线程池
ExecutorService executorService = Executors.newCachedThreadPool();
final ReentrantLockTest reentrantLockTest = new ReentrantLockTest();
executorService.execute(new Runnable() {
@Override
public void run() {
reentrantLockTest.test2();
}
});
executorService.execute(new Runnable() {
@Override
public void run() {
reentrantLockTest.test2();
}
});
执行结果:
Thread[pool-1-thread-1,5,main]得到了锁
Thread[pool-1-thread-2,5,main]获取锁失败
Thread[pool-1-thread-1,5,main]释放了锁
2.1.3、tryLock(long time, TimeUnit unit)
【示例】
//新增方法
public void test3() throws InterruptedException {
if(lock.tryLock(10,TimeUnit.SECONDS))//获取锁
{
try {
System.out.println(Thread.currentThread() + "得到了锁");
TimeUnit.SECONDS.sleep(5);
} catch (Exception e) {
System.out.println("发生异常");
} finally {
System.out.println(Thread.currentThread() + "释放了锁");
lock.unlock();
}
}else{
System.out.println(Thread.currentThread() + "获取锁失败");
}
}
//线程池
ExecutorService executorService = Executors.newCachedThreadPool();
final ReentrantLockTest reentrantLockTest = new ReentrantLockTest();
executorService.execute(new Runnable() {
@Override
public void run() {
try {
reentrantLockTest.test3();
}catch(InterruptedException ie){
System.out.println("发生异常");
}
}
});
executorService.execute(new Runnable() {
@Override
public void run() {
try {
reentrantLockTest.test3();
}catch(InterruptedException ie){
System.out.println("发生异常");
}
}
});
执行结果:
Thread[pool-1-thread-1,5,main]得到了锁
Thread[pool-1-thread-1,5,main]释放了锁
Thread[pool-1-thread-2,5,main]得到了锁
Thread[pool-1-thread-2,5,main]释放了锁
尝试10s获取锁,获取不到才会返回false
2.1.4、lockInterruptibly()
【示例】
//新增方法
public void test4() throws InterruptedException {
lock.lockInterruptibly();
try {
System.out.println(Thread.currentThread()+"得到了锁");
TimeUnit.SECONDS.sleep(10);
} catch (Exception e) {
System.out.println("中断成功");
}finally {
System.out.println(Thread.currentThread()+"释放了锁");
lock.unlock();
}
}
final ReentrantLockTest reentrantLockTest = new ReentrantLockTest();
Thread thread1 = new Thread(new Runnable() {
@Override
public void run() {
try {
reentrantLockTest.test4();
}catch(InterruptedException ie){
System.out.println("发生异常");
}
}
},"1");
thread1.start();
try {
TimeUnit.SECONDS.sleep(3);
} catch (Exception e) {
System.out.println("延迟5s");
}
System.out.println("开始中断");
thread1.interrupt();//中断thread1
Thread thread2 = new Thread(new Runnable() {
@Override
public void run() {
try {
reentrantLockTest.test4();
}catch(InterruptedException ie){
System.out.println("发生异常");
}
}
},"2");
thread2.start();
执行结果:
Thread[1,5,main]得到了锁
开始中断
中断成功
Thread[1,5,main]释放了锁
Thread[2,5,main]得到了锁
Thread[2,5,main]释放了锁
成功中断thread1的锁
三、读写锁
3.1、ReadWriteLock
ReadWriteLock是一个接口,只包含两个方法readLock()、writeLock()
3.2、ReentrantReadWriteLock
ReentrantReadWriteLock实现了ReadWriteLock接口,下面主要说一下ReadLock()、WriteLock()两个方法
【示例1】
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.ReentrantReadWriteLock; /**
* Created by jyy on 2018/6/26.
*/
public class ReentrantReadWriteLockTest { private ReentrantReadWriteLock rwl = new ReentrantReadWriteLock(); /**
* 读锁
*/
public void test1(){
rwl.readLock().lock();
try {
System.out.println(Thread.currentThread() + "得到了读锁");
TimeUnit.SECONDS.sleep(5);
} catch (Exception e) {
System.out.println("发生异常");
} finally {
System.out.println(Thread.currentThread() + "释放了读锁");
rwl.readLock().unlock();
}
} /**
* 写锁
*/
public void test2(){
rwl.writeLock().lock();
try {
System.out.println(Thread.currentThread() + "得到了写锁");
TimeUnit.SECONDS.sleep(5);
} catch (Exception e) {
System.out.println("发生异常");
} finally {
System.out.println(Thread.currentThread() + "释放了写锁");
rwl.writeLock().unlock();
}
} }
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; /**
* Created by jyy on 2018/4/19.
*/
public class Test { public static void main(String[] args) throws InterruptedException {
final ReentrantReadWriteLockTest reentrantReadWriteLockTest = new ReentrantReadWriteLockTest();
ExecutorService executorService = Executors.newCachedThreadPool();//连接池
executorService.execute(new Runnable() {
@Override
public void run() {
reentrantReadWriteLockTest.test1();
}
});
executorService.execute(new Runnable() {
@Override
public void run() {
reentrantReadWriteLockTest.test1();
}
});
executorService.execute(new Runnable() {
@Override
public void run() {
reentrantReadWriteLockTest.test1();
}
});
}
}
执行结果:
Thread[pool-1-thread-1,5,main]得到了读锁
Thread[pool-1-thread-2,5,main]得到了读锁
Thread[pool-1-thread-3,5,main]得到了读锁
Thread[pool-1-thread-3,5,main]释放了读锁
Thread[pool-1-thread-1,5,main]释放了读锁
Thread[pool-1-thread-2,5,main]释放了读锁
多个线程都可以获取到读锁
【示例2】
executorService.execute(new Runnable() {
@Override
public void run() {
reentrantReadWriteLockTest.test1();
}
});
executorService.execute(new Runnable() {
@Override
public void run() {
reentrantReadWriteLockTest.test1();
}
});
executorService.execute(new Runnable() {
@Override
public void run() {
reentrantReadWriteLockTest.test2();
}
});
执行结果:
Thread[pool-1-thread-1,5,main]得到了读锁
Thread[pool-1-thread-2,5,main]得到了读锁
Thread[pool-1-thread-2,5,main]释放了读锁
Thread[pool-1-thread-1,5,main]释放了读锁
Thread[pool-1-thread-3,5,main]得到了写锁
Thread[pool-1-thread-3,5,main]释放了写锁
访问写锁时,必须等读锁全部释放
【示例3】
executorService.execute(new Runnable() {
@Override
public void run() {
reentrantReadWriteLockTest.test2();
}
});
executorService.execute(new Runnable() {
@Override
public void run() {
reentrantReadWriteLockTest.test1();
}
});
executorService.execute(new Runnable() {
@Override
public void run() {
reentrantReadWriteLockTest.test2();
}
});
执行结果:
Thread[pool-1-thread-1,5,main]得到了写锁
Thread[pool-1-thread-1,5,main]释放了写锁
Thread[pool-1-thread-2,5,main]得到了读锁
Thread[pool-1-thread-2,5,main]释放了读锁
Thread[pool-1-thread-3,5,main]得到了写锁
Thread[pool-1-thread-3,5,main]释放了写锁
想要获取读锁,必须先等到写锁释放
总结:
1、如果一个线程占用了读锁,其他线程依然可以获取到这个读锁,但不可以获取到写锁
2、如果一个线程占用了写锁,其他线程不可以获取到读锁和写锁
四、synchronized与lock的区别
1)synchronized是java的一个关键字,而Lock是一个接口
2)synchronized发生异常,会自动释放占有的锁,而Lock必须要主动释放锁,否则会一直处于占用状态
3)Lock中的lockInterruptibly()可以响应中断,而synchronized不可以
4)Lock中的tryLock()可以尝试获取锁,判断是否成功获取到锁,而synchronized不可以
5)ReentrantReadWriteLock可以声明读写锁,提高查询效率
两者不是非一即二的存在,具体使用要看不同的场景
五、锁类型
可重入锁:在执行对象中所有同步方法不用再次获得锁
举个简单的例子,当一个线程执行到某个synchronized方法时,比如说method1,而在method1中会调用另外一个synchronized方法method2,此时线程不必重新去申请锁,而是可以直接执行方法method2
synchronized和ReentrantLock都是可重入锁
可中断锁:在等待获取锁过程中可中断,如ReentrantLock中的lockInterruptibly()
公平锁: 按等待获取锁的线程的等待时间进行获取,等待时间长的具有优先获取锁权利
ReentrantLock默认是非公平锁,可以通过new ReentrantLock(true)声明为公平锁
读写锁:对资源读取和写入的时候拆分为两部分处理,读的时候可以多线程一起读,写的时候必须同步地写,如ReentrantReadWriteLock
JAVA基础知识|synchronized和lock的更多相关文章
- java基础: synchronized与Lock的区别
主要区别 1. 锁机制不一样:synchronized是java内置关键字,是在JVM层面实现的,系统会监控锁的释放与否,lock是JDK代码实现的,需要手动释放,在finally块中释放.可以采用非 ...
- Java基础知识回顾之七 ----- 总结篇
前言 在之前Java基础知识回顾中,我们回顾了基础数据类型.修饰符和String.三大特性.集合.多线程和IO.本篇文章则对之前学过的知识进行总结.除了简单的复习之外,还会增加一些相应的理解. 基础数 ...
- Java基础知识总结(超级经典)
Java基础知识总结(超级经典) 写代码: 1,明确需求.我要做什么? 2,分析思路.我要怎么做?1,2,3. 3,确定步骤.每一个思路部分用到哪些语句,方法,和对象. 4,代码实现.用具体的java ...
- 毕向东—Java基础知识总结(超级经典)
Java基础知识总结(超级经典) 写代码: 1,明确需求.我要做什么? 2,分析思路.我要怎么做?1,2,3. 3,确定步骤.每一个思路部分用到哪些语句,方法,和对象. 4,代码实现.用具体的java ...
- 沉淀,再出发:Java基础知识汇总
沉淀,再出发:Java基础知识汇总 一.前言 不管走得多远,基础知识是最重要的,这些知识就是建造一座座高楼大厦的基石和钢筋水泥.对于Java这门包含了编程方方面面的语言,有着太多的基础知识了,从最初的 ...
- java基础知识一览(二)
一.java基础知识 1.一个文件中只能有一个public的类,因为他的类名要求和文件名相同. 2.classpath变量可以设置其它目录下的类. 例如:类文件所在目录是:F:\Javajdk,那么没 ...
- 黑马毕向东Java基础知识总结
Java基础知识总结(超级经典) 转自:百度文库 黑马毕向东JAVA基础总结笔记 侵删! 写代码: 1,明确需求.我要做什么? 2,分析思路.我要怎么做?1,2,3. 3,确定步骤.每一个思路部 ...
- java基础知识小总结【转】
java基础知识小总结 在一个独立的原始程序里,只能有一个 public 类,却可以有许多 non-public 类.此外,若是在一个 Java 程序中没有一个类是 public,那么该 Java 程 ...
- Java 基础知识(一)
Java基础知识篇: 一.关键字解释 1. final:修饰非抽象类,非抽象方法和属性, 以及修饰方法参数,代表“无法改变的”.出于对设计或者效率的考虑使用该关键字. final类无法被继承,fina ...
随机推荐
- C++复制构造函数,类型转换构造函数,析构函数,引用,指针常量和常量指针
复制构造函数形如className :: className(const &) / className :: className(const className &)后者能以常 ...
- tree 树形加载及增删改
//异步1<template> <div class="addequipment org"> <div class="top"&g ...
- Modelsim问题集锦
前言 收集工程调试中遇到的modelsim问题. 问题 (1)仿真发现时钟信号和理论上的数据信号没有边沿对齐. 解决:一般是时钟精度不匹配的问题. 如果想要1ns的精度则代码中的精度需设置为: v语法 ...
- Jlink调试S5PV210
安装CDT C/C++ Development Toolkit,使eclipse可以开发C/C++项目 Help–>Install New Software中输入:http://download ...
- nodejs 模块全局安装路径配置
nodejs下载安装完成后 输入npm config ls 或者npm config list npm 默认的全局安装路径为该路径,将包都下载在C盘中不是我们想要的结果.一般建议修改在nodejs的安 ...
- JMeter测试clickhouse
使用JMeter对clickhouse连接测试 1.测试计划 jmeter通过JDBC连接数据库需要先引入对应的驱动包,驱动包的版本要与服务器数据库版本一致,我用的驱动版本是:clickhouse-j ...
- 【leetcode】338 .Counting Bits
原题 Given a non negative integer number num. For every numbers i in the range 0 ≤ i ≤ num calculate t ...
- Linux命令——finger
简介 查询并显示系统用户的相关信息. 最小化安装Linux可能没有改名了,需要单独安装. RHEL/CentOS yum install finger* -y Ubuntu apt-get inst ...
- MyBatis_[tp_48]_动态sql_内置参数_parameter&_databaseId
笔记要点-----内置参数_parameter&_databaseId 用处: 迅速切换数据库,执行一条多分支的sql语句即可;1.定义接口 public interface Emp ...
- MyBatsi学习
深入浅出Mybatis系列(一)---Mybatis入门 深入浅出Mybatis系列(二)---配置简介(mybatis源码篇) 深入浅出Mybatis系列(三)---配置详解之properties与 ...