死锁
2021/9/5 6:08:14
本文主要是介绍死锁,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!
死锁
多个线程各自占有一些共享资源﹐并且互相等待其他线程占有的资源才能运行﹐而导致两个或者多个线程都在等待对方释放资源﹐都停止执行的情形.某一个同步块同时拥有“两个以上对象的锁”时,就可能会发生“死锁”的问题.
package com.zishi.Lock; import com.sun.xml.internal.ws.api.model.wsdl.WSDLOutput; //死锁:多个线程互相抱着对方需要的资源,然后形成僵持 public class DeadLock { public static void main(String[] args) { Makeup g1 = new Makeup(0,"灰姑娘"); Makeup g2 = new Makeup(1,"白雪公主"); g1.start(); g2.start(); } } //口红 class Lipstick{ } //镜子 class Mirror{ } class Makeup extends Thread{ //需要的资源只有一份,用static来保证只有一份 static Lipstick lipstick = new Lipstick(); static Mirror mirror = new Mirror(); int choice;//选择 String girlName;//使用化妆品的人 Makeup(int choice,String girlName){ this.choice = choice; this.girlName = girlName; } @Override public void run() { //化妆 try { makeup(); } catch (Exception e) { e.printStackTrace(); } } //化妆,互相持有对方的锁,就是需要拿到对方的资源 private void makeup() throws Exception{ if (choice == 0){ synchronized (lipstick){//获得口号的锁 System.out.println(this.girlName+"获得口红的锁"); Thread.sleep(1000); synchronized(mirror){//一秒后想要获得镜子 System.out.println(this.girlName+"获得镜子的锁"); } } }else { synchronized (mirror){//获得镜子的锁 System.out.println(this.girlName+"获得镜子的锁"); Thread.sleep(2000); synchronized(lipstick){//二秒后想要获得口红 System.out.println(this.girlName+"获得口红的锁"); } } } } }
if (choice == 0){ synchronized (lipstick){//获得口号的锁 System.out.println(this.girlName+"获得口红的锁"); Thread.sleep(1000); synchronized(mirror){//一秒后想要获得镜子 System.out.println(this.girlName+"获得镜子的锁"); } } }else { synchronized (mirror){//获得镜子的锁 System.out.println(this.girlName+"获得镜子的锁"); Thread.sleep(2000); } synchronized(lipstick){//二秒后想要获得口红 System.out.println(this.girlName+"获得口红的锁"); } }
产生死锁的四个必要条件: 1.互斥条件:一个资源每次只能被一个进程使用。
2.请求与保持条件:一个进程因请求资源而阻塞时,对已获得的资源保持不放。
3.不剥夺条件:进程已获得的资源,在末使用完之前,不能强行剥夺。
4.循环等待条件:若干进程之间形成一种头尾相接的循环等待资源关系。
上面列出了死锁的四个必要条件,我们只要想办法破其中的任意一个或多个条件就可以避免死锁发生
Lock锁
package com.zishi.Lock; import java.util.concurrent.locks.ReentrantLock; //测试Lock锁 public class TestLock { public static void main(String[] args) { TestLock2 testLock2 = new TestLock2(); new Thread(testLock2,"小黄").start(); new Thread(testLock2,"老黄").start(); new Thread(testLock2,"黄牛").start(); } } class TestLock2 implements Runnable{ int ticketNums = 10; //定义Lock锁 private final ReentrantLock lock = new ReentrantLock(); @Override public void run() { while (true){ lock.lock();//加锁 try{ lock.lock(); if (ticketNums>0){ try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(Thread.currentThread().getName()+":"+ticketNums--); }else { break; } }finally { //解锁 lock.unlock(); } } } }
这篇关于死锁的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!
- 2024-11-28MQ底层原理资料详解:新手入门教程
- 2024-11-28MQ项目开发资料详解:新手入门教程
- 2024-11-28MQ项目开发资料详解:入门与初级用户指南
- 2024-11-28MQ消息队列资料入门教程
- 2024-11-28MQ消息队列资料:新手入门详解
- 2024-11-28MQ消息中间件资料详解与应用教程
- 2024-11-28MQ消息中间件资料入门教程
- 2024-11-28MQ源码资料详解与入门教程
- 2024-11-28MQ源码资料入门教程
- 2024-11-28RocketMQ底层原理资料详解