前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >面试题之死锁解密

面试题之死锁解密

作者头像
猿天地
发布2018-07-25 17:28:47
6750
发布2018-07-25 17:28:47
举报
文章被收录于专栏:猿天地猿天地

死锁的概念

在多线程环境中,我们经常会遇到多个线程访问同一个共享资源的情况,这个时候必须考虑如何维护数据一致性,常见的方式是加锁处理。只有拿到锁的线程才可以访问共享资源,通过锁就可以让线程对共享资源的访问都是顺序的,避免出现一些数据不一致的问题。

在使用锁的过程中同样也有风险,最为常见的就是死锁现象。死锁就相当于绳子打死结一样,解不开了。在程序中出现这样的情况往往是由于多个线程同时锁住多个资源不释放导致的。

编写一个死锁程序

代码语言:javascript
复制
public class Deadlock {
    public static String str1 = "str1";
    public static String str2 = "str2";

    public static void main(String[] args){
        Thread a = new Thread(() -> {
             try{
                 while(true){
                     synchronized(Deadlock.str1){
                         System.out.println(Thread.currentThread().getName()+"锁住 str1");
                         Thread.sleep(1000);
                         synchronized(Deadlock.str2){
                             System.out.println(Thread.currentThread().getName()+"锁住 str2");
                         }
                     }
                 }
             }catch(Exception e){
                 e.printStackTrace();
             }
        });

        Thread b = new Thread(() -> {
            try{
                while(true){
                    synchronized(Deadlock.str2){
                        System.out.println(Thread.currentThread().getName()+"锁住 str2");
                        Thread.sleep(1000);
                        synchronized(Deadlock.str1){
                             System.out.println(Thread.currentThread().getName()+"锁住 str1");
                        }
                    }
                }
            }catch(Exception e){
                e.printStackTrace();
            }
        });

        a.start();
        b.start();
    }    
}

上面的代码就是一个完整的死锁程序,程序中有两个线程,线程1锁住了str1,获得锁之后休眠1秒钟,这个时候线程2锁住了str2,也进行休眠操作。

线程1休眠完了之后去锁str2,但是str2已经被线程2给锁住了,这边只能等待,同样的道理,线程2休眠完之后也要去锁str1,同样也会等待,这样死锁就产生了。

如果我们将下面这2行代码的值改成一样,死锁还会存在吗?

代码语言:javascript
复制
 public static String str1 = "str1";
 public static String str2 = "str1";

答案是不会,为什么? 网上看到别人的回答:在声明一个对象作为锁的时候要注意字符串类型锁对象,因为字符串有一个常量池,如果不同的线程持有的锁是具有相同字符的字符串锁时,两个锁实际上同一个锁。

如何查看死锁

首先程序不会往下执行了,这是直观的能够看到的死锁现象,看不到的我们可以通过jstack PID查看线程信息,有死锁的话在最下面会告诉我们检测到了死锁的存在,如下图:

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2018-07-09,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 猿天地 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 死锁的概念
  • 编写一个死锁程序
  • 如何查看死锁
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档