死锁 避免死锁的常见方法

简介:

死锁

索是一个非常有用的工具,运用场景非常多,因为它使用起来非常简单,而且易于理解。但同时它也会带来一些困扰,那就是可能会引起死锁,一旦产生死锁,就会造成系统功能不可用。让我们先来看一段代码,这段代码会引起死锁,使线程 thread_1 和线程 thread_2 互相等待对方释放锁。

package thread;

public class DeadLockDemo {
	private static String A = "A";
	private static String B = "B";

	public static void main(String args[]) {
		new DeadLockDemo().deadLock();
	}

	private void deadLock() {

		// 线程thread_1
		Thread thread_1 = new Thread(new Runnable() {

			@Override
			public void run() {
				synchronized (A) {
					System.err.println("--thread_1  lock  A----");
					synchronized (B) {
						System.err.println("--thread_1  lock  B----");
					}
				}
			}
		});

		// 线程thread_2
		Thread thread_2 = new Thread(new Runnable() {

			@Override
			public void run() {
				synchronized (B) {
					System.out.println("--thread_2  lock  B----");
					synchronized (A) {
						System.out.println("--thread_2  lock  A----");
					}
				}
			}
		});

		thread_1.start();
		thread_2.start();
	}
}


这段代码只是演示死锁的场景,在现实中你可能不会写出这样的代码。但是在一些更为复杂的场景中,你可能会遇到这样的问题,比如 thread_1 拿到索之后,因为一些异常情况没有释放索(死循环)。又或者是 thread_1 拿到一个数据库索,释放锁的时候抛出了异常,没释放掉。

一旦出现死锁,业务是可感知的,因为不能继续提供服务了,那么只能通过dump 线程查看到底是哪个线程出现了问题,以下线程信息告诉我们是 DeadLockDemo类的第 35 行和21行引起了死锁。


"Thread-1" prio=6 tid=0x000000000cb13800 nid=0x19ac waiting for monitor entry [0
x000000000d67f000]
   java.lang.Thread.State: BLOCKED (on object monitor)
        at thread.DeadLockDemo$2.run(DeadLockDemo.java:35)
        - waiting to lock <0x00000007d5a9be88> (a java.lang.String)
        - locked <0x00000007d5a9beb8> (a java.lang.String)
        at java.lang.Thread.run(Unknown Source)

"Thread-0" prio=6 tid=0x000000000cb0e800 nid=0x6bc waiting for monitor entry [0x
000000000d48f000]
   java.lang.Thread.State: BLOCKED (on object monitor)
        at thread.DeadLockDemo$1.run(DeadLockDemo.java:21)
        - waiting to lock <0x00000007d5a9beb8> (a java.lang.String)
        - locked <0x00000007d5a9be88> (a java.lang.String)
        at java.lang.Thread.run(Unknown Source)


避免死锁的几个常见方法。

  • 避免一个线程同时获取多个锁。
  • 避免一个线程在索内同时占用多个资源,尽量保证每个索只占用一个资源。
  • 尝试使用定时索,使用 lock.tryLock(timeout) 来替代使用内部索机制。
  • 对于数据库索,加锁和解锁必须在一个数据库连接里,否则会出现解锁失败的情况。

推荐文章:http://www.cnblogs.com/nexiyi/p/java_thread_jstack.html

推荐文章:http://blog.sina.com.cn/s/blog_855eab95010157uv.html


相关文章
|
监控 网络安全 索引
Elasticsearch-通过Kibana查看索引数据
引言   当数据存储到Elasticsearch后,我们希望能方便的通过界面进行查询,有两个工具能够满足我们的需要,一个是Elasticsearch-head插件,另一个是Kibana,笔者认为两个工具各有千秋,大家可以自行体会,不过就安装步骤来说,Elasticsearch-head真心麻烦,本文主要介绍如何部署Kibana,并使用Kibana来查看Elasticsearch中的索引数据。
10818 1
|
SQL 存储 Oracle
6 张图带你彻底搞懂分布式事务 XA 模式
XA 协议是由 X/Open 组织提出的分布式事务处理规范,主要定义了事务管理器 TM 和局部资源管理器 RM 之间的接口。目前主流的数据库,比如 oracle、DB2 都是支持 XA 协议的。
14034 1
6 张图带你彻底搞懂分布式事务 XA 模式
|
7月前
|
Prometheus 监控 Cloud Native
Prometheus+Grafana(docker安装)
本文档详细介绍了如何使用Docker容器快速部署Prometheus监控系统和Grafana数据可视化平台。该方案适用于需要快速搭建监控环境的开发测试场景,具备部署简单、资源占用低、易于维护等特点。
|
存储 缓存 前端开发
Java八股文面试之多线程篇
Java八股文面试之多线程篇
522 0
Java八股文面试之多线程篇
|
搜索推荐 JavaScript Java
Elasticsearch 8.X 如何依据 Nested 嵌套类型的某个字段进行排序?
Elasticsearch 8.X 如何依据 Nested 嵌套类型的某个字段进行排序?
|
Cloud Native Go 数据安全/隐私保护
自定义Docker镜像推送到Docker Hub实战
自定义Docker镜像推送到Docker Hub实战
460 2
自定义Docker镜像推送到Docker Hub实战
|
Docker 容器
Docker安装及镜像源修改
本文介绍了Docker的安装过程和如何修改Docker镜像源以加速下载。包括更新系统包、安装所需软件包、设置yum源、安装Docker以及验证安装是否成功。接着,提供了修改Docker镜像源的步骤,包括创建配置文件、编辑配置文件以设置镜像加速地址,并提供了几个常用的国内镜像源地址。最后,通过重启Docker服务和检查配置是否生效来完成镜像源的修改。
Docker安装及镜像源修改
|
安全 Java 编译器
Go语言面试宝典:50道必会题目与精解
本文提供了50道覆盖Go语言核心概念、并发编程、内存管理、包管理、错误处理和测试等方面的面试题及其详细答案,旨在帮助开发者全面准备Go语言技术面试。
|
缓存 Kubernetes NoSQL
k8s的docker环境下 , redis批量删除key
k8s的docker环境下 , redis批量删除key
607 0
k8s的docker环境下 , redis批量删除key
|
算法 数据库 存储
概要设计与详细设计的区别
概要设计与详细设计的区别     概要设计就是设计软件的结构,包括组成模块,模块的层次结构,模块的调用关系,每个模块的功能等等。同时,还要设计该项目的应用系统的总体数据结构和数据库结构,即应用系统要存储什么数据,这些数据是什么样的结构,它们之间有什么关系。
13457 0
下一篇
oss云网关配置