1-------------------------------------------------------------
在JAVA中实现定时器功能要用的二个类是Timer,TimerTask
Timer类是用来执行任务的类,它接受一个TimerTask做参数
Timer有两种执行任务的模式,最常用的是schedule,它可以以两种方式执行任务:1:在某个时间(Data),2:在某个固定的时间之后(int delay).这两种方式都可以指定任务执行的频率,本文有二个例子,一个是简单的
一个是用了内部类
1.简单实例
先写一个类
public class TimeTest {
public static void main(String[] args) {
Timer timer = new Timer();
timer.schedule(new MyTask(),1000,2000);
}
然后再写个类
public class MyTask extends TimerTask{
@Override
public void run() {
System.out.println("开始运行");
}
}
这样就可以完成一个简单的定时器,但是还有一种方法就是把这二个类写入到一个类中,也就是内部类了
2.内部类
public class SerchRun {
protected static void startRun(){
Timer timer = new Timer();
TimerTask task =new TimerTask(){
public void run(){
System.out.println("开始运行"); //在这写你要调用的方法
}
};
timer.scheduleAtFixedRate(task, new Date(),2000);//当前时间开始起动 每次间隔2秒再启动
// timer.scheduleAtFixedRate(task, 1000,2000); // 1秒后启动 每次间隔2秒再启动
}
public static void main(String[] args) {
SerchRun.startRun();
}
}
schedule和scheduleAtFixedRate的区别在于,如果指定开始执行的时间在当前系统运行时间之前,scheduleAtFixedRate会把已经过去的时间也作为周期执行,而schedule不会把过去的时间算上。
比如:
SimpleDateFormat fTime = new SimpleDateFormat("yyyy/MM/dd HH:mm:ss");
Date d1 = fTime.parse("2005/12/30 14:10:00");
t.scheduleAtFixedRate(new TimerTask(){
public void run()
{
System.out.println("this is task you do6");
}
},d1,3*60*1000);
间隔时间是3分钟,指定开始时间是2005/12/30 14:10:00,如果我在14:17:00分执行这个程序,那么会立刻打印3次
this is task you do6 //14:10
this is task you do6 //14:13
this is task you do6 //14:16
并且注意,下一次执行是在14:19 而不是 14:20。就是说是从指定的开始时间开始计时,而不是从执行时间开始计时。
但是上面如果用schedule方法,间隔时间是3分钟,指定开始时间是2005/12/30 14:10:00,那么在14:17:00分执行这个程序,则立即执行程序一次。并且下一次的执行时间是 14:20,而不是从14:10开始算的周期(14:19)。
2-------------------------------------------------------------
在很多情况下,我们都需要用到定时器
比如:定时删除数据库的某些垃圾信息(即没有用到,但是占着位置的东西)
在CMS中,我们用定时器来定时发布文章,定时删除上传的没有用到的文件…..
定时器,顾名思义一下,在规定的时间提醒某件事情。
从这种解释中,我们不难理解,定时器需要用到的东西有两样:1,定时;2,任务(执行的事件)
先来解决以下第一个问题:定时
在java编程中,用于定时的类有:Timer
在这个类中,有很多方法,最常用的是schedule方法,在这个类中重载了好几个这个方法,但是主要的用途还是用于在某个时间点开始,定时执行某个任务,这个方法保证了定时器整体功能的实现
第二个问题:任务
有了时间,我们就要给这个机器一个任务,让它在规定的时间点后执行这个任务,这个任务类可以继承TimerTask类来实现,这样的话,就可以得到一个新的任务类,这个任务类主要覆盖一个run()方法,这个方法里面我们写的是需要执行的具体任务,这个类在被引用的时候就会执行这个方法。
下面看方法:
任务类:
import java.util.*;
public class Task extends TimerTask {
public void run() {
System.out.println("我是Task"+"现在时间是:" + new Date());
}
}
定时类:
import java.util.Timer;
import java.util.Calendar;
import java.util.GregorianCalendar;
import java.util.Date;
public class Dingshiqi {
public static void main(String[] args) {
Timer timer = new Timer();
Task task = new Task();
timer.schedule(task, new Date(), 2000);
}
}
这样实现的效果是,每隔2秒,就会输出“我是Task,现在时间是(当前的时间)”。
定时器,我们可以用在servlet中,这个是比较常用的,下面说下如何运用servlet实现定时效果
把某些属性值写在xml文件中,然后在servlet中取得该值,然后在servlet的init方法调用定时功能,最后,destroy()来销毁定时器
xml中设置一些相应处理的servlet:
<servlet>
<description>This is the description of my J2EE component</description>
<display-name>This is the display name of my J2EE component</display-name>
<servlet-name>Admin_servlet</servlet-name>
<servlet-class>com.cbh.lmodules_article.servlet.Admin_servlet</servlet-class>
<init-param>
<param-name>startTask</param-name>
<param-value>true</param-value>
</init-param>
<init-param>
<param-name>intervalTime</param-name>
<param-value>1</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
servlet中的init方法启动定时器
public void init() throws ServletException {
ServletContext context = getServletContext();
// (true为用定时间刷新缓存)
String startTask =getInitParameter("startTask");
// 定时刷新时间(分钟)
//Long delay = Long.parseLong(getInitParameter("intervalTime"));
// 启动定时器
if (startTask.equals("true")) {
timer1 = new Timer(true);
task1 = new Timer_task(context);
System.out.println("TimerServlet2定时器已启动");
timer1.schedule(task1,0, 60*1000);
System.out.println("TimerServlet2已经添加任务调度表");
}
}
servlet中的destroy方法销毁定时器
public void destroy() {
super.destroy();
if (timer1 != null) {
timer1.cancel();
System.out.println("TimerServlet2定时器已销毁");
}
}
在servlet中运用定时器的话,那么就需要建立和任务的关联,怎么建立关联呢?
public class Timer_task extends TimerTask {
private ServletContext context;
private static boolean isRunning = false;
public Timer_task(ServletContext context){
this.context = context; //有参构造来实现与servlet的关联
}
Admin_service as=new Admin_service();
@Override
public void run() {
as.delte("");
}
}
3-------------------------------------------------------------
java.util.Timer
一种线程设施,用于安排以后在后台线程中执行的任务。可安排任务执行一次,或者定期重复执行。此类是线程安全的:多个线程可以共享单个 Timer 对象而无需进行外部同步。
java.util.TimerTask
由 Timer 安排为一次执行或重复执行的任务。
如何通过这两个类来完成定时操作功能呢?具体是使用方法,下面通过一个具体demo来进行说明。
本demo实现这样一个功能,在系统运行期间,每30分钟,系统自动检查连接池中的可用连接数,并输出到日志中。
首先创建一个需要定时执行的任务类,这个任务类需要继承TimerTask,然后重写run()方法,run()方法体中的代码就是定时需要执行的操作,在本demo中,就是获取连接池中当前可用连接数,并输出到日志中,具体实现代码如下:
publicclass TaskAvailableConnectNumber extends TimerTask {
private Logger log = Logger.getLogger(TaskAvailableConnectNumber.class); private ConnectionPool pool=ConnectionPool.getInstance(); @Override publicvoid run() { log.debug("当前连接池中可用连接数"+pool.getAvailableConnectNumber()); } } |
下面定义一个监听器,负责在应用服务器启动时打开定时器,监听器需要实现ServletContextListener接口,并重写其中的contextInitialized()和contextDestroyed()方法,代码如下:
publicclass OnLineListener implements ServletContextListener{
private Logger log = Logger.getLogger(OnLineListener.class); Timer timer = null; //在应用服务器启动时,会执行该方法 publicvoid contextInitialized(ServletContextEvent arg0) {
//创建一个定时器,用于安排需要定时执行的任务。 timer = new Timer(); //为定时器安排需要定时执行的任务,该任务就是前面创建的任务类TaskAvailableConnectNumber,并指定该任务每30分钟执行一次。 timer.schedule(new TaskAvailableConnectNumber(), 0, 30*60*1000); log.debug("启动定时器"); } //应用服务器关闭时,会执行该方法,完成关闭定时器的操作。 publicvoid contextDestroyed(ServletContextEvent arg0) { if(timer!=null){ timer.cancel();//关闭定时器 log.debug("-----定时器销毁--------"); } } } |
监听器要想正常运行,需要在web.xml文件中进行配置,配置信息如下:
<!-- 监听器配置开始 --> <listener> <listener-class> cn.sdfi.listen.OnLineListener </listener-class> </listener> <!-- 监听器配置结束 --> |
以上步骤完成后,一个简单的定时器就算开发完成了