活跃性(死锁、饥饿、活锁)

2024-02-17 06:30
文章标签 死锁 饥饿 活锁 活跃性

本文主要是介绍活跃性(死锁、饥饿、活锁),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

锁顺序死锁

我们知道,加锁是为了保证线程安全性而做的同步操作,而过度的加锁可能导致各个线程彼此依赖别的线程已经持有的锁。先上一段代码瞅瞅:

public class DeadLockDemo {

   public static void main(String[] args) {
       Object lock1 = new Object();
       Object lock2 = new Object();

       new Thread(new Runnable() {

           @Override
           public void run() {

               while (true) {
                   synchronized (lock1) {
                       System.out.println("线程t1获取了 lock1锁");
                       LockUtil.sleep(1000L);

                       synchronized (lock2) {
                           System.out.println("线程t1获取了 lock2锁");
                       }
                   }
               }
           }
       }, "t1").start();

       new Thread(new Runnable() {
           @Override
           public void run() {

               while (true) {
                   synchronized (lock2) {
                       System.out.println("线程t2获取了 lock2锁");
                       LockUtil.sleep(1000L);

                       synchronized (lock1) {
                           System.out.println("线程t2获取了 lock1锁");
                       }
                   }
               }
           }
       }, "t2").start();
   }
}

其中的线程t1先获取lock1,再获取lock2,线程t2先获取lock2,再获取lock1,为了让死锁的效果明显的展现出来,我们在每个线程获得到一个锁之后都休眠1秒钟,最后的执行结果是:

线程t1获取了 lock1锁
线程t2获取了 lock2锁

这个程序的运行时序图可以画成这样:

如图,线程t1需要获得线程t2已经获取的lock2锁才能继续执行,线程t2需要获得线程t1已经获取的lock1锁才能继续执行,而这两个线程谁都不愿意先释放已经获取到的锁,造成的尴尬后果就是两个线程僵死在这里。

上边这种因为多个线程试图以不同的顺序来获得相同的锁而造成的死锁也被称为锁顺序死锁。其实多个线程也可能造成这样的锁顺序死锁情况,比如有5个线程,线程2持有线程1需要的锁,线程3持有线程2需要的锁,线程4持有线程3需要的锁,线程5持有线程4需要的锁,而线程1持有线程5需要的锁,这样的依赖就绕成了一个环,结果就是这5个线程都处在永久等待的状态。

隐藏的死锁情况

上边的例子是为了大家容易理解死锁的概念而提出的,真实世界里的死锁情况可能更难被发现。比如我们看下边这个考试场景,一个老师监考若干名学生,试卷一共有100道题,我们先看一下学生的java代码:

public class Student {

   private Teacher teacher;

   private int process;    //答题进度

   public void setTeacher(Teacher teacher) {
       this.teacher = teacher;
   }

   public synchronized int getProcess() {
       return process;
   }

   public synchronized void setProcess(int process) {
       this.process = process;
       if (process == 100) {
           teacher.studentNotify(this);    //学生答完题,通知老师
       }
   }
}

每个Student对象里都维护一个Teacher对象,字段process代表当前的答题进度,可以调用getProcess来获取当前的答题进度,也可以通过setProcess来设置当前的答题进度,当process的值为100时就意味着完成了考试,可以调用Teacher对象的studentNotify方法来交卷。再看一下Teacher的代码:

import java.util.List;

public class Teacher {

   List<Student> students;

   public void setStudents(List<Student> students) {
       this.students = students;
   }

   public synchronized void studentNotify(Student student) {
       students.remove(student);   //将已完成考试的学生从列表中移除
   }

   public synchronized void getAllStudentStatus() {
       for (Student student : students) {
           System.out.println(student.getProcess());
       }
   }
}

每个Teacher对象里都维护了一个students字段,它代表若干个Student对象。每当有学生交卷调用自己的studentNotify方法时,都把该学生从students列表中删除。另外,Teacher还有一个getAllStudentProcess方法,可以获取某个时刻还在答题的各个学生的答题进度。

这两个类貌似人畜无害,但是确暗藏玄机:

  1. StudentsetProcess方法是同步方法,TeacherstudentNotify也是同步方法,而在setProcess方法中可能调用studentNotify方法。也就是说执行setProcess方法的线程需要先获得Student对象的锁,再获得Teacher对象的锁。

  2. TeachergetAllStudentProcess方法是同步方法,StudentgetProcess方法也是同步方法,而在getAllStudentProcess方法中调用了getProcess方法。也就是说执行getAllStudentProcess方法的线程需要先获得Teacher对象的锁,再获得Student对象的锁。

如果线程t1执行setProcess方法,线程t2执行getAllStudentProcess方法。假设Student对象为sTeacher对象为t,那么一种可能的执行时序就是:

所以这两个线程最终可能造成死锁

Student对象的setProcess里调用了Teacher对象的studentNotify方法,我们就称studentNotify方法是Studnet类的外部方法;同样的,getProcess方法也是Teacher类的外部方法。需要我们注意的是:如果在持有锁的情况下调用了某个外部方法,那么就需要警惕死锁

其他资源死锁

只要出现下边这种情况系统都可能进入死锁状态:每个线程都拥有其他线程需要的资源,同时又等待其他线程已经拥有的资源,并且每个线程在获得全部需要的资源之前不会释放已经拥有的资源

我们说其实本身就是一种线程执行需要获取的资源,任何可以被共享的东西都可以被当作一种资源来对待。一个线程完成任务需要获取多个资源,并且多个资源是互斥的,也就是某个资源在同一时刻只能被一个线程拥有,典型的这种资源就是数据库连接,具体情况就不举例子了,等遇到了再说。

预防死锁的建议

前人已经认真的总结过产生死锁的几个必要条件:

  1. 互斥条件:一个资源每次只能被一个线程使用。

  2. 请求与保持条件:一个线程因请求资源而阻塞时,对已获得的资源保持不放。

  3. 不剥夺条件:线程已获得的资源,在未使用完之前,不能强行剥夺。

  4. 循环等待条件:若干线程之间形成一种头尾相接的循环等待资源关系。

只有这4个条件全部成立,死锁的情况才有可能发生。听清楚了,我说的是才有可能发生。因为一般情况下,一个线程持有资源的时间并不会太长,所以一般并不会发生死锁情况,但是如果并发程度很大,也就是非常多的线程在同时竞争资源,如果这四个条件都成立,那么发生死锁的概率将会很大,重要并且可怕的是:一旦系统进入死锁状态,将无法恢复,只能重新启动系统

不过只要破坏上述4个条件中的任意一个,那么死锁的情况就不会发生,所以在设计代码的时候注意这么几点:

  1. 线程在执行任务的过程中,最好进行开放调用

    如果在调用某个方法的时候不需要持有锁,那么这种调用就称为开放调用。像上边Student类调用外部方法studentNotify的时候就已经持有锁了,所以我们可以这样改写一下:

    public void setProcess(int process) {
       synchronized (this) {
           this.process = process;
       }
           if (process == 100) {
               teacher.studentNotify(this);    //学生答完题,通知老师
           }
    }            

    这样锁只用来保护共享变量,而把对studentNotify的调用改为开放调用,这样就不会有死锁的问题啦。同样的,我们可以这样改写getAllStudentProcess方法:

    public void getAllStudentStatus() {
       List<Student> copyOfStudents;
       synchronized (this) {
           copyOfStudents = new ArrayList(students);
       }
       for (Student student : copyOfStudents) {
           System.out.println(student.getProcess());
       }
    }

    我们把共享变量students在线程本地复制了一份,锁只用来保护复制students的时候不会有别的线程去修改它,从而把对getProcess的调用也改为了开放调用开放调用其实是避免了循环等待条件,从而使死锁不可能发生。

    当然,如果代码没有了锁保护就会丧失原子性,如果某个外部调用需要被加锁保护来和其他一些操作共同组成某个大的原子性操作的话,就不能进行开放调用了~

  2. 各个线程最好用固定的顺序来获取资源。

    我们可以改写一下最开始的那个例子,线程t1改写成这样:

    synchronized (lock1) {
       System.out.println("线程t1获取了 lock1锁");
       LockUtil.sleep(1000L);
       synchronized (lock2) {
           System.out.println("线程t1获取了 lock2锁");
       }
    }

    线程t2改写成这样:

    synchronized (lock1) {
       System.out.println("线程t2获取了 lock1锁");
       LockUtil.sleep(1000L);
       synchronized (lock2) {
           System.out.println("线程t2获取了 lock2锁");
       }
    }

    在一个线程获取lock1锁的时候,另一个线程就不能执行了,只能等待已经获取锁的线程把所有操作都做完后释放锁再执行。也就是说不存在循环等待条件了。

  3. 可以让持有资源的时间有限。

    我们前边用到的synchronized加锁机制是没有超时时间的,也就是说如果一个线程获取锁之后,直到同步代码块中的代码执行完成之后才能释放锁。所以在死锁的情况下,一个线程是不会主动去释放锁的,如果我们让锁有了超时时间,就可以打破不剥夺条件

    比如前边说的十字路哭堵车的例子,如果规定一个车如果30秒不能前行,则先倒车几秒钟后重试,这样就可以破解死锁的魔咒了。

    这个带有超时时间的锁我们在唠叨性能的时候再详细说怎么使用哈~

饥饿

线程饥饿是另一种活跃性问题,也可以使程序无法执行下去。如果一个线程因为处理器时间全部被其他线程抢走而得不到处理器运行时间,这种状态被称之为饥饿,一般是由高优先级线程吞噬所有的低优先级线程的处理器时间引起的。

java语言在Thread类中提供了修改线程优先级的成员方法setPriority,并且定义了10个优先级级别。不同操作系统有不同的线程优先级,java会把这10个级别映射到具体的操作系统线程优先级上边。操作系统的线程调度会按照自己的调度策略来轮番执行我们定义的线程。

我们所设置的线程优先级对操作系统来说只是一种建议,当我们尝试提高一个线程的优先级的时候,可能起不到任何作用,也可能使这个线程过度优先执行,导致别的线程得不到处理器分配的时间片,从而导致饿死。所以我们尽量不要修改线程的优先级,具体效果取决于具体的操作系统,并且可能导致某些线程饿死

小贴士:

我们还可以把处理器想象成皇帝,把各个线程想象成妃子,皇帝隔几分钟就换一个妃子陪他。我们设置线程优先级就像是调整某个妃子的好看程度,具体皇帝挑不挑这个妃子还是具体的皇帝说了算,而且不同的皇帝有不同的口味,最后结果是啥还真说不准。如果我们把一个妃子弄的很好看,一个皇帝太宠信她,从而使某些妃子得不到宠信,就是传说中的`饥饿`现象。

活锁

虽然不会像死锁那样因为获取不到资源而阻塞,也不会像饥饿那样得不到处理器时间而无可奈何,活锁仍旧可以让程序无法执行下去~

比如在一间教室里,狗哥要出去,猫爷要进来,门只能容得下一个人进出,而它们在门口相遇了,所以狗哥往后退了一步意思是猫爷先进,而猫爷也退了一步意思是狗哥先进;之后狗哥往前走了一步,猫爷也往前走了一步,俩人又都堵在了门口,所以又都同时退一步,然后再同时进一步,同时退一步,同时进一步…..

把狗哥和猫爷都比做一个线程的话,这两个线程虽然都没有停止运行,但是却无法向下执行,这种情况就是所谓的活锁

为了解决这个问题,需要在遇到冲突重试时引入一定的随机性。比如狗哥和猫爷在门口相遇都后退时,狗哥隔一秒后再前进,猫爷隔两秒后再前进,这样就不会有同时走到门口的尴尬了~

总结

  1. 一旦系统进入死锁状态,将无法恢复,只能重新启动系统。,产生死锁的4个条件:

  • 互斥条件:一个资源每次只能被一个线程使用。

  • 请求与保持条件:一个线程因请求资源而阻塞时,对已获得的资源保持不放。

  • 不剥夺条件:线程已获得的资源,在未使用完之前,不能强行剥夺。

  • 循环等待条件:若干线程之间形成一种头尾相接的循环等待资源关系。

预防死锁只需要破坏上边四个条件之一就好,下边是一些建议:

  • 线程在执行任务的过程中,最好进行开放调用。

  • 各个线程最好用固定的顺序来获取资源。

  • 可以让持有资源的时间有限。

如果一个线程因为处理器时间全部被其他线程抢走而得不到处理器运行时间,这种状态被称之为饥饿,一般是由高优先级线程吞噬所有的低优先级线程的处理器时间引起的。

活锁虽然不会像死锁那样因为获取不到资源而阻塞,也不会像饥饿那样得不到处理器时间而无可奈何,活锁仍旧可以让程序无法执行下去,最好在遇到冲突重试时引入一定的随机性。

这篇关于活跃性(死锁、饥饿、活锁)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/716938

相关文章

【Linux修行路】线程安全和死锁

目录 ⛳️推荐 一、线程安全 1.1 常见的线程不安全情况 1.2 常见的线程安全情况 1.3 常见的不可重入情况 1.4 常见可重入的情况 1.5 可重入与线程安全的联系 1.6 可重入与线程安全的区别 二、死锁 2.1 死锁的四个必要条件 2.2 如何避免产生死锁? ⛳️推荐 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大

模拟线程死锁——Thread学习笔记

记录一下之前写过的一段模拟死锁的代码: /*** 模拟死锁** @author lixiang* @date 2018年10月12日 - 9:51* @history 2018年10月12日 - 9:51 lixiang create.*/public class HoldLockDemo {private static Object[] lock = new Object[10];priv

【编程底层思考】如何检测和避免线程死锁

一、什么是线程死锁? 线程死锁发生在多个线程因为争夺资源而相互阻塞,导致程序无法正常结束的情况。例如,线程A持有资源2并等待资源1,线程B持有资源1并等待资源2,这样就形成了死锁。 二、如何检测死锁? 使用jmap、jstack等命令行工具查看JVM的线程栈和堆内存情况,jstack可以显示死锁信息。使用VisualVM、JConsole等图形化工具进行排查。例如,JConsole可以连接到

C++11 Thread线程池、死锁、并发

一、线程与进程         进程:运行中的程序         线程:进程中的小进程 二、线程库的使用         包含头文件#include<thread> 2.1 thread函数         具体代码: void show(string str) {cout << "This is my word : " << str << endl;}int main() {t

C++ 的死锁问题的发生和避免

C/C++程序中产生死锁的原因很多,本文大致归纳了下面几类,分别做分析。 1.单线程/进程多次加锁导致死锁 单线程导致死锁的情况一般是由于调用了引起阻塞的函数,比如(copy_from_user()、copy_to_ser()、和kmalloc()),阻塞后进行系统调度,调度的过程中有可能又调用了之前获取锁的函数,这样必然导致死锁。 还有一种就是自旋锁函数在没有释放锁马上又进行申请同一个自旋

yt零售系统订单死锁原因

知识前提: InnoDB引擎在加锁的时候,只有通过索引进行检索的时候才会使用行级锁,否则会使用表级锁。 场景: 在订单服务中,开起事务,对同一张表,先更新(无索引),再新增,发生死锁。 原因: 同一线程,更新事务未提交,因为无索引导致了表锁,再新增的时候当前线程等待更新释放锁,会把当前线程挂起来,而锁正是被自己占用,该线程又被挂起而没机会释放锁。 解决方法: 更新的时候在检索列创建索

Oracle:杀死死锁进程

Oracle:杀死死锁进程 1. 模拟死锁现象 利用PL/SQL Developer工具可以很容易模拟死锁现象。用同一个数据库的同一个用户登录2个PL/SQL Developer。 首先,在其中一个PL/SQL Developer随便对数据库的表执行一个更新操作,不要提交,状态为“待提交” 然后,在另一个PL/SQL Developer执行同样的操作,此时这个操作会等待前面的事务提交之后

Android 线程死锁的案例

模拟java 线程死锁 ExecutorService executorService = Executors.newFixedThreadPool(2);Object lockA =new Object();Object lockB =new Object();executorService.submit(new Runnable() {@Overridepublic void

java线程死锁例子及解决方法

Java线程死锁是由于有些资源彼此交叉取用,就有可能造成死结. 如1线程中 取得A对象的锁定后又要取得B对象的锁定.但是同时2线程中取得B对象的锁定后又要取得A对象的锁定.这两个线程同时发生时就会造成,1线程拿到A对象锁定后等待B对象的锁定.2线程拿到B对象锁定后等待A对象锁定.这样就会进入没有停止的等待中. 线程死锁的一个简单例子: package deadLockThread;publi

Lesson_for_java_day19--java的多线程(多线程概念、单例设计模式、死锁)

一、多线程 /*线程由两种实现方式:第一种方式:class MyThread extends Thread{public void run(){需要进行执行的代码,如循环。}}public class TestThread{public static void main(String[] args){Thread t1=new Mythread();T1.start();}}只有等到所有的