基于Java线程实现后台定时监控

2016-02-19 21:40 32 1 收藏

下面图老师小编要向大家介绍下基于Java线程实现后台定时监控,看起来复杂实则是简单的,掌握好技巧就OK,喜欢就赶紧收藏起来吧!

【 tulaoshi.com - 编程语言 】

  http://tailsherry.javaeye.com/blog/176152

  熟悉编写JavaScript的人,都习惯在页面写入setTimeOut来实现web页面的定时监控或事务处理。但是如何在Java服务端来实现这样一个监控机制呢?一般大家都会想到线程。但是一直以来,都没有亲身动手实践过。趁着工作间隙,自己也搬出一段代码来,与大家一起分享线程编程的神奇魔力。

   

  首先创建一个基本抽象类SchedThread,代码内容如下:

  view plaincopy to clipboardprint?
  package com.test;  
   
  /** 
   * 基于Java线程实现后台定时监控 P Created: Mar 26, 2008 10:08:43 /PP 
   * /PH4http://tailsherry.javaeye.com/H4 
   * P 
   *  
   * @author TailSherry 
   */ 
  public abstract class SchedThread  
  {  
      protected static final long NEVER = Long.MAX_VALUE;  
   
      // 定义一个线程锁,保证当前只有一个工作在操作中  
      private final Object lock = new Object();  
   
      // 定义一个Thread变量  
      private Thread thread;  
   
      // 控制线程循环的开关  
      private boolean active = true;  
   
      // 定义一个毫秒级的时间变量,指示何时执行下一个操作  
      private long nextTime;  
   
      /** 
       * 定义个一个抽象的方法用来获取下一个执行操作的时间,可使用NEVER 
       */ 
      protected abstract long getNextTime();  
   
      /** 
       * 定义一个抽象的方法,让子类来定义具体的工作过程 
       */ 
      protected abstract void executeWork();  
   
      protected String getName()  
      {  
          return getClass().getName();  
      }  
   
      /** 
       * 启动线程 
       */ 
      public void start()  
      {  
          thread = new Thread(new Runnable()  
          {  
              public void run()  
              {  
                  runInternal();  
              }  
          }, getName());  
          thread.start();  
      }  
   
      /** 
       * 强迫停止线程,跳出for循环 
       */ 
      public void stop() throws InterruptedException  
      {  
          synchronized (lock)  
          {  
              active = false;  
              lock.notify();  
          }  
          thread.join();  
      }  
   
      /** 
       * 此方法可以在任何时候激活当前线程,让线程进入工作执行环节 
       */ 
      public void workAdded(long time)  
      {  
          synchronized (lock)  
          {  
              if (time nextTime)  
              {  
                  // 立刻激活线程工作继续运行  
                  lock.notify();  
              }  
          }  
      }  
   
      /** 
       * 线程监测控制逻辑部分 
       */ 
      private void runInternal()  
      {  
          // 无限循环  
          for (;;)  
          {  
              // 该过程忽略了所有的Exception,以保证线程不会因此而中断  
              try 
              {  
                  synchronized (lock)  
                  {  
                      nextTime = getNextTime();  
                      // 获得时间区间,即要等待的时间段  
                      long interval = nextTime - System.currentTimeMillis();  
                      if (interval 0)  
                      {  
                          try 
                          {  
                              lock.wait(interval);  
                          }  
                          catch (InterruptedException e)  
                          {  
                              // 忽略此Exception  
                          }  
                      }  
                      // 如果active为false,强制中断  
                      if (!active)  
                      {  
                          break;  
                      }  
                  }  
                  // 执行具体的工作  
                  executeWork();  
              }  
              catch (Throwable t)  
              {  
                  try 
                  {  
                      Thread.sleep(10000);  
                  }  
                  catch (InterruptedException ie)  
                  {  
                      // 忽略此Exception  
                  }  
              }  
          }  
      }  
  }/P 

  package com.test;

  /**
   * 基于Java线程实现后台定时监控  Created: Mar 26, 2008 10:08:43
   * http://tailsherry.javaeye.com
   *
   *
   * @author TailSherry
   */
  public abstract class SchedThread
  {
      protected static final long NEVER = Long.MAX_VALUE;

      // 定义一个线程锁,保证当前只有一个工作在操作中
      private final Object lock = new Object();

      // 定义一个Thread变量
      private Thread thread;

      // 控制线程循环的开关
      private boolean active = true;

      // 定义一个毫秒级的时间变量,指示何时执行下一个操作
      private long nextTime;

      /**
       * 定义个一个抽象的方法用来获取下一个执行操作的时间,可使用NEVER
       */
      protected abstract long getNextTime();

      /**
       * 定义一个抽象的方法,让子类来定义具体的工作过程
       */
      protected abstract void executeWork();

      protected String getName()
      {
          return getClass().getName();
      }

      /**
       * 启动线程
       */
      public void start()
      {
          thread = new Thread(new Runnable()
          {
              public void run()
              {
                  runInternal();
              }
          }, getName());
          thread.start();
      }

      /**
       * 强迫停止线程,跳出for循环
       */
      public void stop() throws InterruptedException
      {
          synchronized (lock)
          {
              active = false;
              lock.notify();
          }
          thread.join();
      }

      /**
       * 此方法可以在任何时候激活当前线程,让线程进入工作执行环节
       */
      public void workAdded(long time)
      {
          synchronized (lock)
          {
              if (time nextTime)
              {
                  // 立刻激活线程工作继续运行
                  lock.notify();
              }
          }
      }

      /**
       * 线程监测控制逻辑部分
       */
      private void runInternal()
      {
          // 无限循环
          for (;;)
          {
              // 该过程忽略了所有的Exception,以保证线程不会因此而中断
              try
              {
                  synchronized (lock)
                  {
                      nextTime = getNextTime();
                      // 获得时间区间,即要等待的时间段
                      long interval = nextTime - System.currentTimeMillis();
                      if (interval 0)
                      {
                          try
                          {
                              lock.wait(interval);
                          }
                          catch (InterruptedException e)
                          {
                              // 忽略此Exception
                          }
                      }
                      // 如果active为false,强制中断
                      if (!active)
                      {
                          break;
                      }
                  }
                  // 执行具体的工作
                  executeWork();
              }
              catch (Throwable t)
              {
                  try
                  {
                      Thread.sleep(10000);
                  }
                  catch (InterruptedException ie)
                  {
                      // 忽略此Exception
                  }
              }
          }
      }
  }

(本文来源于图老师网站,更多请访问https://www.tulaoshi.com/bianchengyuyan/)

  以上这个类非常关键,基本上已经实现了所有的控制逻辑,如此再扩展出一个实现类出来,比如这里我写了一个模拟实现类MyDataGenerator,大家可以参考一下:

  Java代码 view plaincopy to clipboardprint?
    

   view plaincopy to clipboardprint?
  package com.test;     
      
  public class MyDataGenerator extends SchedThread {     
      protected void executeWork() {     
          System.out.println("Execute work ...");     
      }     
      
      protected long getNextTime() {     
          return System.currentTimeMillis() + 2000L;     
      }     
      
      public static void main(String argv[]) {     
          MyDataGenerator generator = new MyDataGenerator();     
          generator.start();     
      }     
  }   

  package com.test;  
   
  public class MyDataGenerator extends SchedThread {  
      protected void executeWork() {  
          System.out.println("Execute work ...");  
      }  
   
      protected long getNextTime() {  
          return System.currentTimeMillis() + 2000L;  
      }  
   
      public static void main(String argv[]) {  
          MyDataGenerator generator = new MyDataGenerator();  
          generator.start();  
      }  
  } 
  当然这里没有使用workAdded和stop等功能,可以留给大家扩展。
  

(本文来源于图老师网站,更多请访问https://www.tulaoshi.com/bianchengyuyan/)

来源:https://www.tulaoshi.com/n/20160219/1626816.html

延伸阅读
标签: Delphi
一、把当前进程变为一个系统服务: 目的是在任务列表中把程序隐藏起来。调用API函数RegisterServiceProcess实现。 二、定义全局热键(本例中定义热键Ctrl+Del+R),步骤: 1、定义捕获Windows消息WM_HOTKEY的钩子函数,即:procedure WMHotKey(var Msg : TWMHotKey); message WM_HOTKEY; 2、向Windows加入一...
---- 摘要:在Java出现之前,编写多线程程序是一件烦琐且伴随许多不安全因素的事情。利用Java,编写安全高效的多线程程序变得简单,而且利用多线程和Java的网络包我们可以方便的实现多线程服务器程序。 ---- Java是伴随Internet的大潮产生的,对网络及多线程具有内在的支持,具有网络时代编程语言的一切特点。从Java的当前应用看...
在Java 语言中,提供了各种各样的输入输出流(stream),使我们能够很方便的对数据进行操作,其中,管道(pipe)流是一种非凡的流,用于在不同线程(threads)间直接传送数据。一个线程发送数据到输出管道,另一个线程从输入管道中读数据。通过使用管道,实现不同线程间的通讯。!-- frame contents -- !-- /frame contents --无需求助于类似临...
代码如下: import org.apache.log4j.*; public class TaskJob {        public static Logger log = Logger                      .getLogger(TaskJob.class);        p...
原理是使用LinkedHashMap来实现,当缓存超过大小时,将会删除最老的一个元组。 实现代码如下所示 代码如下: import java.util.LinkedHashMap; import java.util.Map; public class LRUCache {  public static class CachedData {   private Object data = null;   private long time = 0;   private boole...

经验教程

567

收藏

70
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部