java多线程中死锁情况的一个示例
2015-01-04 10:18
274 查看
下面是死锁情况的一个示例代码
[java] view
plaincopy
![](https://code.csdn.net/assets/CODE_ico.png)
![](https://code.csdn.net/assets/ico_fork.svg)
package com.qust.demo.money;
class A {
public synchronized void foo(B b) {
System.out.println(Thread.currentThread().getName() + " 进入A的foo");
try {
Thread.sleep(200);
} catch (InterruptedException ex) {
ex.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + " 试图调用B的last");
b.last();
}
public synchronized void last() {
System.out.println("A的last()");
}
}
class B {
public synchronized void bar(A a) {
System.out.println(Thread.currentThread().getName() + " 进入B的bar");
try {
Thread.sleep(200);
} catch (InterruptedException ex) {
ex.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + " 试图调用A的last");
a.last();
}
public synchronized void last() {
System.out.println("B的last()");
}
}
public class DeadLock implements Runnable {
A a = new A();
B b = new B();
public void init() {
Thread.currentThread().setName("主线程");
System.out.println("进入主线程");
a.foo(b);
}
public void run() {
Thread.currentThread().setName("副线程");
System.out.println("进入副线程");
b.bar(a);
}
public static void main(String[] args) {
DeadLock dl = new DeadLock();
new Thread(dl).start();
dl.init();
}
}
下面是运行结果
[html] view
plaincopy
![](https://code.csdn.net/assets/CODE_ico.png)
![](https://code.csdn.net/assets/ico_fork.svg)
进入主线程
进入副线程
主线程 进入A的foo
副线程 进入B的bar
副线程 试图调用A的last
主线程 试图调用B的last
我们看到,正常情况下,最后还应该打印出“A的last()”和"B的last()",但是因为线程死锁的原因,所以程序一直在等待执行,没能正常的执行下去。
在上面的代码里面,为什么会出现死锁这种情况呢?我们来简单分析一下。
首先虽然副线程的start()是在主线程的init()之前,但是因为多线程开启也需要一段时间,所以我们可以看到,是主线程的init()方法执行在前,然后在init()里面调用了a.foo()。A类和B类中的方法都是同步方法,因此,A的对象和B的对象都是同步锁。在进入A的foo()之前,线程会对A加锁,然后线程睡眠200ms,这时候副线程调用B的bar(),同样的会对B加锁,然后睡眠200ms。这时候,主线程唤醒,试图调用B的bar方法,因为是同步方法,所以需要对B加锁,但是这时候B已经被副线程锁住了,所以主线程就一直处于阻塞状态。当副线程唤醒的时候,试图调用A的同步方法,同样需要对A加锁,但是这时候主线程持有A的锁,并处于阻塞状态,所以副线程也不能向下执行,大家都在等着对方释放锁,因此出现了我们上面说的死锁的情况。
如果我们把A和B的last()的synchronized去掉,那么程序在调用last()之前就不需要对对象进行加锁,也就不会出现死锁的情况。
因为在工作中还没有遇到这种情况,所以只能拿这个实例程序讲解了。
[java] view
plaincopy
![](https://code.csdn.net/assets/CODE_ico.png)
package com.qust.demo.money;
class A {
public synchronized void foo(B b) {
System.out.println(Thread.currentThread().getName() + " 进入A的foo");
try {
Thread.sleep(200);
} catch (InterruptedException ex) {
ex.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + " 试图调用B的last");
b.last();
}
public synchronized void last() {
System.out.println("A的last()");
}
}
class B {
public synchronized void bar(A a) {
System.out.println(Thread.currentThread().getName() + " 进入B的bar");
try {
Thread.sleep(200);
} catch (InterruptedException ex) {
ex.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + " 试图调用A的last");
a.last();
}
public synchronized void last() {
System.out.println("B的last()");
}
}
public class DeadLock implements Runnable {
A a = new A();
B b = new B();
public void init() {
Thread.currentThread().setName("主线程");
System.out.println("进入主线程");
a.foo(b);
}
public void run() {
Thread.currentThread().setName("副线程");
System.out.println("进入副线程");
b.bar(a);
}
public static void main(String[] args) {
DeadLock dl = new DeadLock();
new Thread(dl).start();
dl.init();
}
}
下面是运行结果
[html] view
plaincopy
![](https://code.csdn.net/assets/CODE_ico.png)
进入主线程
进入副线程
主线程 进入A的foo
副线程 进入B的bar
副线程 试图调用A的last
主线程 试图调用B的last
我们看到,正常情况下,最后还应该打印出“A的last()”和"B的last()",但是因为线程死锁的原因,所以程序一直在等待执行,没能正常的执行下去。
在上面的代码里面,为什么会出现死锁这种情况呢?我们来简单分析一下。
首先虽然副线程的start()是在主线程的init()之前,但是因为多线程开启也需要一段时间,所以我们可以看到,是主线程的init()方法执行在前,然后在init()里面调用了a.foo()。A类和B类中的方法都是同步方法,因此,A的对象和B的对象都是同步锁。在进入A的foo()之前,线程会对A加锁,然后线程睡眠200ms,这时候副线程调用B的bar(),同样的会对B加锁,然后睡眠200ms。这时候,主线程唤醒,试图调用B的bar方法,因为是同步方法,所以需要对B加锁,但是这时候B已经被副线程锁住了,所以主线程就一直处于阻塞状态。当副线程唤醒的时候,试图调用A的同步方法,同样需要对A加锁,但是这时候主线程持有A的锁,并处于阻塞状态,所以副线程也不能向下执行,大家都在等着对方释放锁,因此出现了我们上面说的死锁的情况。
如果我们把A和B的last()的synchronized去掉,那么程序在调用last()之前就不需要对对象进行加锁,也就不会出现死锁的情况。
因为在工作中还没有遇到这种情况,所以只能拿这个实例程序讲解了。
相关文章推荐
- 一个简单的Java死锁示例
- 【代码示例】 一个简单的Java死锁
- java多线程发生死锁的一个例子
- 一个javamemcache客户端,多线程情况下很强悍
- 一个简单的java死锁示例
- [java]一个多线程引起的死锁(class初始化,锁)
- JAVA套接字(Socket)101七天系列—第五天【一个多线程的示例】
- java多线程中死锁情况的一个示例
- c++多线程之死锁的发生的情况解析(包含两个归纳,6个示例)
- c++多线程之死锁的发生的情况解析(包含两个归纳,6个示例)
- 0038 Java学习笔记-多线程-传统线程间通信、Condition、阻塞队列、《疯狂Java讲义 第三版》进程间通信示例代码存在的一个问题
- 一个简单的Java死锁示例(转)
- java学习--多线程发生死锁的一个例子
- 关于Java中多线程死锁的代码示例
- Java第七课 Java的多线程程序进程和线程的概念,实现多线程的两种方式,线程同步的原理,线程的死锁,运用wait和notify来实现producer - consumer关系,线程终止的两种情况。
- 多线程Java Socket编程示例(http://www.blogjava.net/sterning/archive/2007/10/15/152940.html)
- 借用一个程序 来说明 多线程 和单态 以及 对象在两种情况下的存储方式。
- 使用.bat文件运行Java程序的一个示例
- java写的一个死锁例子
- 简析 java多线程中的死锁与生命性