Java线程意外退出线程后自动重启

最近有个简单需求,创建了一个线程用来循环读取队列中的数据,然后存到数据库,一切美好都源自于想象。殊不知线程总是莫名其妙的死亡。所以决定用观察者模式去监听线程;

前置知识

Observable与Observer

  • 在Java中通过Observable类和Observer接口实现了观察者模式。一个Observer对象监视着一个Observable对象的变化,当Observable对象发生变化时,Observer得到通知,就可以进行相应的工作。例如在文档/视图结构中,文档被修改了,视图就会得到通知。

  • java.util.Observable中有两个方法对Observer特别重要,一个是setChange()方法用来设置一个内部标志位注明数据发生了变化;一个是notifyObservers()方法会去调用一个列表中所有的Observer的update()方法,通知它们数据发生了变化。

  • Observer通过Observable的addObserver()方法把自己添加到这个列表中。这个列表虽然由Observable拥有,但Observable并不知道到底有哪些Observer正在观察等待通知。Observable只提供一个方法让Observer能把自己添加进列表,并保证会去通知Observer发生了变化。通过这种机制,可以有任意多个Observer对Observable进行观察,而不影响Observable的实现。

被监听的类

import cn.hutool.core.date.DateUtil;
import java.util.Observable;

public  class RunThread1 extends Observable implements Runnable{

    private String name;
    public RunThread1(String name){
        this.name=name;
    }
     // 此方法一经调用,立马可以通知观察者,在本例中是监听线程
     public void doBusiness(){
         if ( true ){
             super .setChanged();
         }
         notifyObservers();
     }
     @Override
     public void run() {
         int c =  0 ;
         while ( true ){     //模拟线程运行一段时间之后退出
             System.out.println( "Runing- " +c+ " "+name + DateUtil.now());
             try {
                 Thread.sleep( 2000 );
             } catch  (InterruptedException e) {
                 e.printStackTrace();
                 doBusiness();
                 break ;
             }
             c++;
             //模拟抛出异常
             try {
                 if (c==  4 ){
                     String str =  null ;
                     str.length(); //此处将会抛出空指针异常
                 }
             } catch  (Exception e) {
                 e.printStackTrace();
                 doBusiness(); //在抛出异常时调用,通知观察者,让其重启线程
                 break ; //异常抛出之后,一定要跳出循环,保证将线程送进地狱
             }
         }    
     }
} 

观察者类

import  java.util.Observable;
import  java.util.Observer;
public  class Listener1 implements  Observer{
    private String name;
    public Listener1(String name){
        this.name=name;
    };
     @Override
     public void update(Observable o, Object arg) {
         System.out.println( "RunThread死机" );
         RunThread1 run =  new RunThread1(name);
         run.addObserver( this );
         new Thread(run).start();
         System.out.println( "RunThread重启" );
     }
}

开始测试

测试代码

     public static  void  main(String[] args) {
        String name="lasse";
         RunThread1 run =  new RunThread1(name);
         Listener1 listen =  new Listener1(name);
         run.addObserver(listen);
         new Thread(run).start();
         //run.doBusiness();
     }

测试结果

Runing- 0 lasse2023-03-21 11:32:08
Runing- 1 lasse2023-03-21 11:32:10
Runing- 2 lasse2023-03-21 11:32:12
Runing- 3 lasse2023-03-21 11:32:14
java.lang.NullPointerException
at com.bcy.adaptor.starter.RunThread1.run(RunThread1.java:36)
at java.lang.Thread.run(Thread.java:748)
RunThread死机
RunThread重启
Runing- 0 lasse2023-03-21 11:32:16
Runing- 1 lasse2023-03-21 11:32:18
Runing- 2 lasse2023-03-21 11:32:20
Runing- 3 lasse2023-03-21 11:32:22
java.lang.NullPointerException
at com.bcy.adaptor.starter.RunThread1.run(RunThread1.java:36)
at java.lang.Thread.run(Thread.java:748)
RunThread死机
RunThread重启
Runing- 0 lasse2023-03-21 11:32:24
Runing- 1 lasse2023-03-21 11:32:26
Runing- 2 lasse2023-03-21 11:32:28
Runing- 3 lasse2023-03-21 11:32:30
java.lang.NullPointerException
at com.bcy.adaptor.starter.RunThread1.run(RunThread1.java:36)
at java.lang.Thread.run(Thread.java:748)
RunThread死机
RunThread重启
。。。。。。

  • 3
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值