JUC之多线程锁问题
多线程锁
8种问题锁状态:
该部分全部围绕的是以下内容并结合相应的例子:synchronized实现同步的基础:Java中每个对象都可以作为锁。
具体表现为以下三种形式:(之前只是简单的了解)
- 对于普通同步方法,锁是当前实例对象。
- 对于静态同步方法,锁是当前类的Class对象。
- 对于同步方法块,锁是Synchonized括号里配置的对象
当一个线程试图访问同步代码块时,它首先必须得到锁,退出或抛出异常时必须释放锁
也就是说如果一个实例对象的普通同步方法获取锁后,该实例对象的其他普通方法必须等待获取锁的方法释放锁后才能获取锁
,可是别的实例对象的非静态同步方法因为跟该实例对象的普通同步方法用的是不同的锁,所以必须等待该实例对象已获取锁的普通同步方法释放锁就可以获取他们自己的锁。
所以的静态同步方法用的也是同一把锁---类对象本身,这两把锁(this/class)是不同的对象,所以静态同步方法与非静态同步方法之间是不会有竞争条件的。但是一旦一个静态同步方法获取锁后,其他的静态同步方法都必须等待该方法释放锁后才能获得锁,而不管是同一实例对象的静态同步方法之间,还是不同的实例对象的静态同步方法之间,只要它们是同一个类的实例对象。
具体的例子如下:
package com.JUC;
import java.util.concurrent.TimeUnit;
class phone{
public static synchronized void sendEmail() throws Exception {
// Thread.sleep(4000);
//暂定4s
TimeUnit.SECONDS.sleep(1);
System.out.println("sendEmail-------");
}
public synchronized void sendMessage() throws Exception {
System.out.println("sendSMS----------");
}
public void hello(){
System.out.println("hello wold");
}
}
public class lockPhenomenon {
public static void main(String[] args) throws InterruptedException {
//phone 是模板--class,new phone --this
phone phone = new phone();
phone phone2 = new phone();
new Thread(()->{
try {
phone.sendEmail();
} catch (Exception e) {
e.printStackTrace();
}
},"AAA").start();
Thread.sleep(100);
new Thread(()->{
try {
// phone.sendMessage();
//phone.hello();
phone2.sendMessage();
} catch (Exception e) {
e.printStackTrace();
}
},"BBB").start();
}
}
/**
- 多线程8锁
- 标准访问,先打印邮件还是短信
- sendEmail-------
- sendSMS----------
- 2、邮件方法暂停4s,请问先打印邮件还是短信
- sendEmail-------
- sendSMS----------
- 解释1-2问题:synchronized锁的是当前类对象,一个对象里面如果有多个synchronized方法,在某个时刻内,只要一个线程去调用其中的一个synch方法了,其他线程都得只能等待,换句话说,某一个时刻内,只能有唯一一个线程去访问这些synchronized方法,锁的是当前对象this,被锁定后,其他的线程都不能进入到当前对象的其他synchronized方法。
- 3、新增一个普通方法hello,先打印邮件还是hello.
- hello wold
- sendEmail-------
解释:加个普通方法后发现和同步锁无关,- 4、两部手机,先打印邮件还是先打印短信。
- sendSMS----------
- sendEmail-------
- 解释4:换成两个对象后,不是同一把锁了,情况立刻变化
- 5、两个静态同步方法,同一部手机,请问先打印邮件还是短信
- sendEmail-------
- sendSMS----------
- 6、两个静态同步方法,两部手机,请问先打印邮件还是短信
- sendEmail-------
- sendSMS----------
- 解释5-6问题:都换成静态同步方法后,情况又发生了变化,synchronized锁的是new--this,具体的对象(如一部部手机)
*static synchronized锁的是 静态 class---模板- 7、一个普通同步方法,一个静态同步方法,1部手机,请问先打印邮件还是短信
- sendSMS----------
- sendEmail-------
- 8、一个普通同步方法,一个静态同步方法,2部手机,请问先打印邮件还是短信
- sendSMS----------
- sendEmail-------
*/
重入锁:
重进入是指任意线程在获取到锁之后能够再次获取该锁而不会被锁所阻塞,并且支持获取锁时的公平和非公平性选择
该特性的实现需要解决以下两个问题。
1)线程再次获取锁。锁需要去识别获取锁的线程是否为当前占据锁的线程,如果是,则再次成功获取。
2)锁的最终释放。线程重复n次获取了锁,随后在第n次释放该锁后,其他线程能够获取到该锁。锁的最终释放要求锁对于获取进行计数自增,计数表示当前锁被重复获取的次数,而锁被释放时,计数自减,当计数等于0时表示锁已经成功释放
ReentrantLock:显式的重进入,调用lock()方法时,已经获取到锁的线程,能够再次调用lock()方法获取锁而不被阻塞。
synchronized:隐式的重进入,如一个synchronized修饰的递归方法,在方法执行时,执行线程在获取了锁之后仍能连续多次地获得该锁,而不像Mutex由于获取了锁,而在下一次获取锁时出现阻塞自己的情况。
公平锁与非公平锁:
非公平锁:线程饿死,但是效率高
公平锁:雨露均沾,效率相对较低
公平性与否是针对获取锁而言的,如果一个锁是公平的,那么锁的获取顺序就应该符合请求的绝对时间顺序,也就是FIFO(先进先出)。
死锁:
之前的文章内容:Java多线程编程(同步、死锁、生产消费)
线程 A 持有资源 2,线程 B 持有资源 1,他们同时都想申请对方的资源,所以这两个线程就会互相等待而进入死锁状态。
产生死锁必须具备以下四个条件(操作系统部分):
- 互斥条件:该资源任意一个时刻只由一个线程占用。
- 请求与保持条件:一个线程因请求资源而阻塞时,对已获得的资源保持不放。
- 不剥夺条件:线程已获得的资源在末使用完之前不能被其他线程强行剥夺,只有自己使用完毕后才释放资源。
- 循环等待条件:若干线程之间形成一种头尾相接的循环等待资源关系
解决方式:(破坏其中之一就好)
破坏互斥条件(无法破坏)
破坏请求与保持条件
破坏不剥夺条件
破坏循环等待条件
·避免一个线程同时获取多个锁。
·避免一个线程在锁内同时占用多个资源,尽量保证每个锁只占用一个资源。
·尝试使用定时锁,使用lock.tryLock(timeout)来替代使用内部锁机制。
·对于数据库锁,加锁和解锁必须在一个数据库连接里,否则会出现解锁失败的情况。
实现代码:
package com.JUC;
import java.util.concurrent.TimeUnit;
public class deadlock05 {
static Object a = new Object();
static Object b = new Object();
public static void main(String[] args) {
new Thread(()->{
synchronized(a){ //获取a资源,并加锁
System.out.println(Thread.currentThread().getName()+"持有a资源,试图获取b资源");
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
synchronized (b){ //获取b资源,并加锁
System.out.println(Thread.currentThread().getName()+"获取到b资源");
}
}
},"A线程").start();
new Thread(()->{
synchronized(b){ //获取b资源,并加锁
System.out.println(Thread.currentThread().getName()+"持有b资源,试图获取a资源");
synchronized (a){ //获取a资源,并加锁
System.out.println(Thread.currentThread().getName()+"获取到a资源");
}
}
},"B线程").start();
}
}
B线程持有b资源,试图获取a资源
A线程持有a资源,试图获取b资源
验证是否是死锁:
- jps 类似Linux ps -ef
- jstack JVM自带的堆栈跟踪工具
代码在运行的时候,进入命令行:
找到我们正在运行的程序pid.
观察下图:
最后显示发现一个死锁,然后上面的内容解析:
B线程当前锁的是....fb00,等待的是....faf0
A线程当前锁的是....faf0,等待的是....fb00;
JUC之多线程锁问题的更多相关文章
- Java并发编程(3) JUC中的锁
一 前言 前面已经说到JUC中的锁主要是基于AQS实现,而AQS(AQS的内部结构 .AQS的设计与实现)在前面已经简单介绍过了.今天记录下JUC包下的锁是怎么基于AQS上实现的 二 同步锁 同步锁不 ...
- Python多线程锁
[Python之旅]第六篇(四):Python多线程锁 python lock 多线程 多线程使用方法 多线程锁 摘要: 在多线程程序执行过程中,为什么需要给一些线程加锁以及如何加锁,下面就来 ...
- java 并发多线程 锁的分类概念介绍 多线程下篇(二)
接下来对锁的概念再次进行深入的介绍 之前反复的提到锁,通常的理解就是,锁---互斥---同步---阻塞 其实这是常用的独占锁(排它锁)的概念,也是一种简单粗暴的解决方案 抗战电影中,经常出现为了阻止日 ...
- Java多线程--锁的优化
Java多线程--锁的优化 提高锁的性能 减少锁的持有时间 一个线程如果持有锁太长时间,其他线程就必须等待相应的时间,如果有多个线程都在等待该资源,整体性能必然下降.所有有必要减少单个线程持有锁的时间 ...
- synchronized与static synchronized 的差别、synchronized在JVM底层的实现原理及Java多线程锁理解
本Blog分为例如以下部分: 第一部分:synchronized与static synchronized 的差别 第二部分:JVM底层又是怎样实现synchronized的 第三部分:Java多线程锁 ...
- Java - "JUC" ReentrantLock获取锁
[Java并发编程实战]-----“J.U.C”:ReentrantLock之一简介 ReentrantLock介绍 ReentrantLock是一个可重入的互斥锁,又被称为“独占锁”. 顾名思义,R ...
- Java - "JUC" ReentrantLock释放锁
Java多线程系列--“JUC锁”04之 公平锁(二) 释放公平锁(基于JDK1.7.0_40) 1. unlock() unlock()在ReentrantLock.java中实现的,源码如下: p ...
- Java——多线程锁的那些事
引入 Java提供了种类丰富的锁,每种锁因其特性的不同,在适当的场景下能够展现出非常高的效率. 下面先带大家来总体预览一下锁的分类图 1.乐观锁 VS 悲观锁 乐观锁与悲观锁是一种广义上的概念,体现了 ...
- JUC(三):JUC包下锁概念
线程不安全集合类 ArrayList List是线程不安全的集合类,底层是Object数组实现,初始化容量是10(其实是一个空数组,第一次扩容时,将数组扩容为10),其后每次扩容大小为当前容量的一半( ...
随机推荐
- 超过三张表禁止join
一. 问题提出 <阿里巴巴JAVA开发手册>里面写超过三张表禁止join,这是为什么? 二.问题分析 对这个结论,你是否有怀疑呢?也不知道是哪位先哲说的不要人云亦云,今天我设计sql,来验 ...
- spring切面-单线程简单权限判定
spring切面简单模拟用户权限判定 需求: 游客:仅注册用户 用户:修改,注册 管理员:删除,查询,修改,注册 1,文件配置 导包 src下创建applicationContext.xml文件配置如 ...
- PDF.js Electron Viewer
PDF.js 是基于 HTML5 解析与渲染 PDF 的 JavaScript 库,由 Mozilla 主导开源. 本文旨在介绍 PDF.js 于 Electron 里如何开始使用,实际尝试了用其 A ...
- JUC之Lock接口以及Synchronized回顾
Lock接口 Synchronized关键字回顾: 多线程编程步骤(上): 创建资源类,在资源类创建属性和操作方法 创建多个线程,调用资源类的操作方法 创建线程的四种方式: 继承Thread 实现Ru ...
- pwnable_start
第一次接触这种类型的题,例行检查一下 题目是32位 没有开启nx保护可以通过shellocode来获得shell 将题目让如ida中 由于第一次碰到这种题,所以我会介绍的详细一点, 可以看到程序中调用 ...
- [BUUCTF]PWN——hitcontraining_magicheap
hitcontraining_magicheap 附件 步骤: 例行检查,64位程序,开启了nx和canary 本地试运行一下,经典的堆的菜单 64位ida载入,检索程序里的字符串的时候发现了后门 m ...
- 网络编程socket 结合IO多路复用select; epool机制分别实现单线程并发TCP服务器
select版-TCP服务器 1. select 原理 在多路复用的模型中,比较常用的有select模型和epoll模型.这两个都是系统接口,由操作系统提供.当然,Python的select模块进行了 ...
- 我写了个IDEA开源插件,vo2dto 一键生成对象转换
让人头疼的对象转换 头炸,po2vo.vo2do.do2dto,一堆对象属性,取出来塞进来.要不是为了 DDD 架构下的各个分层防腐,真想一竿子怼下去. 那上 BeanUtils.copyProper ...
- 分享自己做的一个指定进程以及线程长时间cpu监控的工具
前言: 前面给大家分享过一个工作中用到的编译拷贝脚本,其实工作中还有一些其他工具的使用,今天再来分享一个自己纯手工的CPU监控的脚本.大家可以结合上篇文章与本篇文章一起学习shell. 主要实现功能: ...
- centos使用docker安装tomcat8
下载镜像 docker pull tomcat:8 启动 docker run -d -p 8080:8080 -v /data/tomcat/webapps/:/usr/local/tomcat/w ...