多线程锁

8种问题锁状态:

该部分全部围绕的是以下内容并结合相应的例子:synchronized实现同步的基础:Java中每个对象都可以作为锁。

具体表现为以下三种形式:(之前只是简单的了解)

  1. 对于普通同步方法,锁是当前实例对象。
  2. 对于静态同步方法,锁是当前类的Class对象。
  3. 对于同步方法块,锁是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锁
    1. 标准访问,先打印邮件还是短信
  • 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,他们同时都想申请对方的资源,所以这两个线程就会互相等待而进入死锁状态。

产生死锁必须具备以下四个条件(操作系统部分):

  1. 互斥条件:该资源任意一个时刻只由一个线程占用。
  2. 请求与保持条件:一个线程因请求资源而阻塞时,对已获得的资源保持不放。
  3. 不剥夺条件:线程已获得的资源在末使用完之前不能被其他线程强行剥夺,只有自己使用完毕后才释放资源。
  4. 循环等待条件:若干线程之间形成一种头尾相接的循环等待资源关系

解决方式:(破坏其中之一就好)

  1. 破坏互斥条件(无法破坏)

  2. 破坏请求与保持条件

  3. 破坏不剥夺条件

  4. 破坏循环等待条件

·避免一个线程同时获取多个锁。

·避免一个线程在锁内同时占用多个资源,尽量保证每个锁只占用一个资源。

·尝试使用定时锁,使用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资源

验证是否是死锁:

  1. jps 类似Linux ps -ef
  2. jstack JVM自带的堆栈跟踪工具

代码在运行的时候,进入命令行:

找到我们正在运行的程序pid.

观察下图:

最后显示发现一个死锁,然后上面的内容解析:

B线程当前锁的是....fb00,等待的是....faf0

A线程当前锁的是....faf0,等待的是....fb00;

JUC之多线程锁问题的更多相关文章

  1. Java并发编程(3) JUC中的锁

    一 前言 前面已经说到JUC中的锁主要是基于AQS实现,而AQS(AQS的内部结构 .AQS的设计与实现)在前面已经简单介绍过了.今天记录下JUC包下的锁是怎么基于AQS上实现的 二 同步锁 同步锁不 ...

  2. Python多线程锁

    [Python之旅]第六篇(四):Python多线程锁   python lock 多线程 多线程使用方法 多线程锁 摘要:   在多线程程序执行过程中,为什么需要给一些线程加锁以及如何加锁,下面就来 ...

  3. java 并发多线程 锁的分类概念介绍 多线程下篇(二)

    接下来对锁的概念再次进行深入的介绍 之前反复的提到锁,通常的理解就是,锁---互斥---同步---阻塞 其实这是常用的独占锁(排它锁)的概念,也是一种简单粗暴的解决方案 抗战电影中,经常出现为了阻止日 ...

  4. Java多线程--锁的优化

    Java多线程--锁的优化 提高锁的性能 减少锁的持有时间 一个线程如果持有锁太长时间,其他线程就必须等待相应的时间,如果有多个线程都在等待该资源,整体性能必然下降.所有有必要减少单个线程持有锁的时间 ...

  5. synchronized与static synchronized 的差别、synchronized在JVM底层的实现原理及Java多线程锁理解

    本Blog分为例如以下部分: 第一部分:synchronized与static synchronized 的差别 第二部分:JVM底层又是怎样实现synchronized的 第三部分:Java多线程锁 ...

  6. Java - "JUC" ReentrantLock获取锁

    [Java并发编程实战]-----“J.U.C”:ReentrantLock之一简介 ReentrantLock介绍 ReentrantLock是一个可重入的互斥锁,又被称为“独占锁”. 顾名思义,R ...

  7. Java - "JUC" ReentrantLock释放锁

    Java多线程系列--“JUC锁”04之 公平锁(二) 释放公平锁(基于JDK1.7.0_40) 1. unlock() unlock()在ReentrantLock.java中实现的,源码如下: p ...

  8. Java——多线程锁的那些事

    引入 Java提供了种类丰富的锁,每种锁因其特性的不同,在适当的场景下能够展现出非常高的效率. 下面先带大家来总体预览一下锁的分类图 1.乐观锁 VS 悲观锁 乐观锁与悲观锁是一种广义上的概念,体现了 ...

  9. JUC(三):JUC包下锁概念

    线程不安全集合类 ArrayList List是线程不安全的集合类,底层是Object数组实现,初始化容量是10(其实是一个空数组,第一次扩容时,将数组扩容为10),其后每次扩容大小为当前容量的一半( ...

随机推荐

  1. go 代理

    环境变量中设置 #GO111MODULE=auto GOPROXY=https://goproxy.io 如果不第一次,则在命令行设置 go env -w GO111MODULE=on go env ...

  2. ebs 初始化登陆

    BEGIN fnd_global.APPS_INITIALIZE(user_id => youruesr_id, esp_id => yourresp_id, resp_appl_id = ...

  3. 实现nfs持久挂载+autofs自动挂载

    实验环境: 两台主机 node4:192.168.37.44 NFS服务器 node2:192.168.37.22 客户端 在nfs服务器,先安装nfs和rpcbind [root@node4 fen ...

  4. Activiti工作流引擎使用详解(一)

    一.IDEA安装activiti插件 在插件库中查找actiBPM,安装该插件,如果找不到该插件,请到插件库中下载该包手动安装,插件地址 http://plugins.jetbrains.com/pl ...

  5. apply 和 call 的区别

    相同点: 都能够改变方法的执行上下文(执行环境),将一个对象的方法交给另一个对象来执行,并且是立即执行 不同点: call方法从第二个参数开始可以接收任意个参数,每个参数会映射到相应位置的func的参 ...

  6. NTLM验证过程

    中我们介绍Kerberos认证的整个流程.在允许的环境下,Kerberos是首选的认证方式.在这之前,Windows主要采用另一种认证协议 --NTLM(NT Lan Manager).NTLM使用在 ...

  7. Axios的正确食用方法

    这里分享出我个人封装的一个axios,我会尽量每行注释,希望对大家有所帮助. 1. 安装 全局执行代码 npm install axios; 2. 编写全局axios文件(附件里有代码) 在src目录 ...

  8. C语言程序设计:模拟简单运算器的工作

    目录 C语言程序设计:模拟简单运算器的工作 1.题目 2.分析 3.代码实现 4.结尾 C语言程序设计:模拟简单运算器的工作 1.题目 ​ 模拟简单运算器的工作,输入一个算式(没有空格),遇等号&qu ...

  9. int i=i++;和i=++i;和i++

    1.int i=i++; 2.i=++i; 3.i++

  10. 痞子衡嵌入式:揭秘i.MXRT1170上用J-Link连接复位后PC总是停在0x223104的原因

    大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是i.MXRT1170上安全调试策略实现对JLink调试的影响. 痞子衡之前写过一篇旧文 <i.MXRT600的ISP模式下用J-L ...