Java多线程——线程的死锁
Java多线程——线程的死锁
摘要:本文主要介绍了Java多线程中遇到的死锁问题。
部分内容来自以下博客:
https://www.cnblogs.com/wy697495/p/9757982.html
https://www.cnblogs.com/maydow/p/4899110.html
https://www.cnblogs.com/digdeep/p/4448148.html
死锁的产生
产生原因
多个线程同时被阻塞,它们中的一个或者全部都在等待某个资源被释放,而该资源又被其他线程锁定,从而导致每一个线程都得等其它线程释放其锁定的资源,造成了所有线程都无法正常结束。
必要条件
这是从网上其他文档看到的死锁产生的四个必要条件:
◆ 互斥使用,即当资源被一个线程使用(占有)时,别的线程不能使用。
◆ 不可抢占,资源请求者不能强制从资源占有者手中夺取资源,资源只能由资源占有者主动释放。
◆ 请求和保持,即当资源请求者在请求其他的资源的同时保持对原有资源的占有。
◆ 循环等待,即存在一个等待队列:P1占有P2的资源,P2占有P3的资源,P3占有P1的资源。这样就形成了一个等待环路。
当上述四个条件都成立的时候,便形成死锁。当然,死锁的情况下如果打破上述任何一个条件,便可让死锁消失。
死锁案例
下面展示了一个死锁的代码案例,线程A拿到了资源A需要获取资源B,线程B拿到了资源B需要获取资源A,当两个线程都在等待资源时,就出现了死锁。
public class Demo {
public static void main(String[] args) {
DemoThread a = new DemoThread("A", "线程A");
DemoThread b = new DemoThread("B", "线程B");
a.start();
b.start();
}
}
class Resource {
public static Object A = new Object();
public static Object B = new Object();
}
class DemoThread extends Thread {
private String sign;
public DemoThread(String sign, String name) {
this.sign = sign;
this.setName(name);
}
@Override
public void run() {
if ("A".equals(sign)) {
HasA();
}
if ("B".equals(sign)) {
HasB();
}
}
public void HasA() {
synchronized (Resource.A) {
System.out.println(Thread.currentThread().getName() + " >>> " + "Has A, Need B");
synchronized (Resource.B) {
System.out.println(Thread.currentThread().getName() + " >>> " + "Has A & Has B");
}
}
}
public void HasB() {
synchronized (Resource.B) {
System.out.println(Thread.currentThread().getName() + " >>> " + "Has B, Need A");
synchronized (Resource.A) {
System.out.println(Thread.currentThread().getName() + " >>> " + "Has B & Has A");
}
}
}
}
当出现死锁时,控制台打印结果如下。
1 线程A >>> Has A, Need B
2 线程B >>> Has B, Need A
避免死锁
避免嵌套封锁:这是死锁最主要的原因的,如果你已经有一个资源了就要避免封锁另一个资源。如果你运行时只有一个对象封锁,那是几乎不可能出现一个死锁局面的。
只对有请求的进行封锁:你应当只对你要使用的资源进行加锁。
避免无限期的等待:如果两个线程正在等待对象结束,无限期的使用线程加入,如果你的线程必须要等待另一个线程的结束,那最好设置一个等待的最长时间。
Java多线程——线程的死锁的更多相关文章
- Java多线程中的死锁
Java多线程中的死锁 死锁产生的原因 线程死锁是指由两个以上的线程互相持有对方所需要的资源,导致线程处于等待状态,无法往前执行. 当线程进入对象的synchronized代码块时,便占有了资源,直到 ...
- java 多线程—— 线程让步
java 多线程 目录: Java 多线程——基础知识 Java 多线程 —— synchronized关键字 java 多线程——一个定时调度的例子 java 多线程——quartz 定时调度的例子 ...
- java 多线程—— 线程等待与唤醒
java 多线程 目录: Java 多线程——基础知识 Java 多线程 —— synchronized关键字 java 多线程——一个定时调度的例子 java 多线程——quartz 定时调度的例子 ...
- Java多线程--线程及相关的Java API
Java多线程--线程及相关的Java API 线程与进程 进程是线程的容器,程序是指令.数据的组织形式,进程是程序的实体. 一个进程中可以容纳若干个线程,线程是轻量级的进程,是程序执行的最小单位.我 ...
- Java多线程-线程的同步(同步方法)
线程的同步是保证多线程安全访问竞争资源的一种手段.线程的同步是Java多线程编程的难点,往往开发者搞不清楚什么是竞争资源.什么时候需要考虑同步,怎么同步等等问题,当然,这些问题没有很明确的答案,但有些 ...
- Java多线程——线程的优先级和生命周期
Java多线程——线程的优先级和生命周期 摘要:本文主要介绍了线程的优先级以及线程有哪些生命周期. 部分内容来自以下博客: https://www.cnblogs.com/sunddenly/p/41 ...
- Java多线程——线程的创建方式
Java多线程——线程的创建方式 摘要:本文主要学习了线程的创建方式,线程的常用属性和方法,以及线程的几个基本状态. 部分内容来自以下博客: https://www.cnblogs.com/dolph ...
- Java多线程——线程之间的协作
Java多线程——线程之间的协作 摘要:本文主要学习多线程之间是如何协作的,以及如何使用wait()方法与notify()/notifyAll()方法. 部分内容来自以下博客: https://www ...
- Java多线程——线程之间的同步
Java多线程——线程之间的同步 摘要:本文主要学习多线程之间是如何同步的,如何使用volatile关键字,如何使用synchronized修饰的同步代码块和同步方法解决线程安全问题. 部分内容来自以 ...
随机推荐
- poj——1422 Air Raid
Air Raid Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 8577 Accepted: 5127 Descript ...
- flask应用的分页
Flask-SQLAlchemy支持分页 https://www.jianshu.com/p/5e03cd202728
- component and slot
component and slot 使用: 1.component panel <article class="message"> <div class=&qu ...
- Spring Boot中微信全局token的缓存实现
为什么要缓存token? 这里的token指的是微信JSAPI中基础支持的ACCESS_TOKEN,并非网页授权ACCESS_TOKEN.网页授权Token每天的调用次数没有限制,不需要缓存. 接口 ...
- GNS3配置SecureCRT
C:\SecureCRT\SecureCRT.exe /script D:\GNS3\DyRouter.vbs /T /telnet 127.0.0.1 %p "D:\Program Fil ...
- linux驱动之中断方式获取键值
linux驱动之中断方式获取键值 ----------------------------------------------------------------------------------- ...
- DirectX11 学习笔记7 - 支持自由移动的摄像机
如今将又一次制定一个camera摄像机.能够自由移动. 比方前进 后退,上游 下潜. 各个方向渲染之类的. 首先设置按键. 这个时候须要在 XWindow.h 里面 bool XWindow::fra ...
- pagefile.sys
pagefile.sys
- uboot中添加自己的命令【转】
本文转载自:http://blog.csdn.net/huanghai381/article/details/51206646 每个命令都是通过U_BOOT_CMD宏来定义的.这个宏定义了一个相关的结 ...
- YTU 2506: 切面条
2506: 切面条 时间限制: 1 Sec 内存限制: 128 MB 提交: 382 解决: 223 题目描述 一根高筋拉面,中间切一刀,可以得到2根面条. 如果先对折1次,中间切一刀, ...