java 定时器 的几种实现及优缺点

本文探讨了Java中使用Timer进行任务调度的方法,详细解析了其实现过程,并对其优缺点进行了深入分析。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一:java Timer 在 java.util.Timer包中的Timer 类可以实现定时功能。
1. 使用 Timer 进行任务调度

<span style="font-size:18px;">public class TimerTest extends TimerTask { 
public TimerTest(){} </span>
<span style="font-size:18px;">@Override 
 public void run() { 
 System.out.println("定时任务处理逻辑"); 
 } </span>
<span style="font-size:18px;">public static void main(String[] args) { 
        Timer timer = new Timer(); 
        long delay1 = 1 * 1000; 
        long period1 = 1000; 
        // 从现在开始延时 1 秒钟之后开始,每隔 1 秒钟执行一次run 方法。
        timer.schedule(new TimerTest("job1"), delay1, period1); 
        long delay2 = 2 * 1000; 
        long period2 = 2000; 
</span><pre name="code" class="displaycode"><span style="font-size:18px;">        // 从现在开始延时 2秒钟之后开始,每隔 2 秒钟执行一次run 方法。
        timer.schedule(new TimerTest(), delay2, period2); 

        //指定每天的指定时间执行 定时任务(run 方法的逻辑处理)
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        SimpleDateFormat sdf1 = new SimpleDateFormat("yyyy-MM-dd ");
        String nowDate = sdf1.format(new Date());
        String timerTime = "23:49:00";
        Timer timer = new Timer();
        String timerDateStr = nowDate + timerTime;
        Date timerDate = null;
        try {
            timerDate = sdf.parse(timerDateStr);
        } catch (ParseException e) {
            e.printStackTrace();
        }
        //</span><span style="font-size:18px;">timerDate时间小于当前时间则自动执行一次业务逻辑,当前时间未达到指定该参数的时间时,
        //只有达到该参数指定时间时才会执行任务
</span><span style="font-size:18px;">        timer.schedule(new </span><span style="font-size:18px;">TimerTest()</span>,<span style="font-size:18px;"> timerDate);</span>
 } 
<span style="font-size:18px;">优缺:Timer 的优点在于简单易用,但由于所有任务都是由同一个线程来调度,因此所有任务都是串行执行的,
同一时间只能有一个任务在执行,前一个任务的延迟或异常都将会影响到之后的任务。而且当前任务如果抛出
运行时异常则该任务终止,如果<span style="color:#FF0000;">当前任务是每天定时任务</span>则下次时间点的任务则不会执行。</span>

<span style="font-size:18px;">
<strong>二:ScheduledExecutor</strong></span>
<pre name="code" class="displaycode"><span style="font-size:18px;">import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;

public class ScheduledExecutorTest implements Runnable {
	private String jobName = "";

	public ScheduledExecutorTest(String jobName) {
		super();
		this.jobName = jobName;
	}

	@Override
	public void run() {
		System.out.println("execute " + jobName);
	}

	public static void main(String[] args) {
		ScheduledExecutorService service = Executors.newScheduledThreadPool(10);

		long initialDelay1 = 1;
		long period1 = 1;
        // 从现在开始1秒钟之后,每隔1秒钟执行一次job1
		service.scheduleAtFixedRate(
		        new ScheduledExecutorTest("job1"), initialDelay1,
				period1, TimeUnit.SECONDS);

		long initialDelay2 = 1;
		long delay2 = 1;
        // 从现在开始2秒钟之后,每隔2秒钟执行一次job2
		service.scheduleWithFixedDelay(
		        new ScheduledExecutorTest("job2"), initialDelay2,
				delay2, TimeUnit.SECONDS);
</span>
}
<pre name="code" class="displaycode"><pre name="code" class="displaycode"><span style="font-size:18px;"></span><pre name="code" class="displaycode"><pre name="code" class="displaycode"><span style="font-size:18px;">}</span><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><pre name="code" class="displaycode"><span style="font-size:18px;">优缺:ScheduledExecutor:</span>

 
 
 
 
 
 
 
 
 
 
 是java 5 推出了基于线程池设计的 ScheduledExecutor。其设计思想是,每一个被调 
 
 
 
 
 度的任务都会由线程池中一个线程去执行,因此任务是并发执行的,相互之间不会受到干扰。需要注意的是, 

只有当任务的执行时间到来时,ScheduedExecutor 才会真正启动一个线程,其余时间 ScheduledExecutor 都
是在轮询任务的状态。此定时器在抛出运行时异常时 虽然不会影响其他启动的任务但是本次的任务将会退出,
自此后不再执行。


三:基于框架Spring 的定时器:
如果你用了Spring 框架了,那么 本人极力推荐这种定时器,因为他集齐上述两种方式的有点于一身,并且没
有前两者的缺点。即:当前任务中抛出了运行时异常,当前任务线程不会退出。如果程序是每天定时任务,那
么在下一个任务运行的时间点任务仍然继续。

使用Spring 定时器需要的jar包:quartz-all-1.6.0.jar 和 jta.jar两个jar包。

配置文件:
在Spring 的业务逻辑bean容器配置文件中添加如下配置:
<!-- Spring 定时器  start -->
    <bean id="activitytask" class="com.gop.service.activitytimer.ActivityTimer">
        <property name="activityServiceImpl">
            <ref bean="activityServiceImpl"/>
        </property>
        <property name="activityRecordServiceImpl">
            <ref bean="activityRecordServiceImpl"/>
        </property>
        <property name="mailsServiceImpl">
            <ref bean="mailsServiceImpl"/>
        </property>
    </bean>
//如果是一个普通的类,不需要注入dao层
<bean id=" activitytask " class=" com.gop.service.activitytimer.ActivityTimer "></bean>
//上面的两个bean节点根据自己的情况任选一个,下面的bean节点都是需要的。
    <bean id="triggerB" class="org.springframework.scheduling.quartz.CronTriggerBean">
        <property name="jobDetail">
            <ref bean="jobDetailB"/>
        </property>
        <property name="cronExpression">
            <value>0 34 17 * * ?</value>
        </property>
    </bean>
    <bean id="jobDetailB" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
        <property name="targetObject">
            <ref bean="activitytask"/>
        </property>
        <property name="targetMethod" value="activityStart"/> //----①
    </bean>
    <bean id="scheduler" lazy-init="false" class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
        <property name="triggers">
            <list>
                <ref bean="triggerB"/>
            </list>
        </property>
    </bean>

实现类:
com.gop.service.activitytimer.ActivityTimer
public class ActivityTimer {
    public static final SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd");
    public static final SimpleDateFormat sdfTim=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
    private ActivityServiceImpl activityServiceImpl;
    private MailsServiceImpl mailsServiceImpl;
    private ActivityRecordServiceImpl activityRecordServiceImpl;
    public ActivityTimer(){}
    /**
     * 定时器主处理方法
     */
    public void activityStart() { //这里的方法名与配置文件中
处名称一致
        //每天晚24点 处理线上生效的活动.给玩家发奖。
//
activityManage 为自定义处理逻辑方法。
        activityManage();
    }
}


时间的配置如下:
<value>
0 34 17 * * ?< value> 代表每天的17:34:00 开始执行任务

时间大小由小到大排列,从秒开始,顺序为 秒,分,时,天,月,年    *为任意 ?为无限制。由此上面所配置的内容就是,在每天的16点26分启动znrwdb方法


具体时间设定可参考

"0/10 * * * * ?" 每10秒触发

"0 0 12 * * ?" 每天中午12点触发
"0 15 10 ? * *" 每天上午10:15触发
"0 15 10 * * ?" 每天上午10:15触发
"0 15 10 * * ? *" 每天上午10:15触发
"0 15 10 * * ? 2005" 2005年的每天上午10:15触发
"0 * 14 * * ?" 在每天下午2点到下午2:59期间的每1分钟触发
"0 0/5 14 * * ?" 在每天下午2点到下午2:55期间的每5分钟触发
"0 0/5 14,18 * * ?" 在每天下午2点到2:55期间和下午6点到6:55期间的每5分钟触发
"0 0-5 14 * * ?" 在每天下午2点到下午2:05期间的每1分钟触发
"0 10,44 14 ? 3 WED" 每年三月的星期三的下午2:10和2:44触发
"0 15 10 ? * MON-FRI" 周一至周五的上午10:15触发
"0 15 10 15 * ?" 每月15日上午10:15触发
"0 15 10 L * ?" 每月最后一日的上午10:15触发
"0 15 10 ? * 6L" 每月的最后一个星期五上午10:15触发
"0 15 10 ? * 6L 2002-2005" 2002年至2005年的每月的最后一个星期五上午10:15触发
"0 15 10 ? * 6#3" 每月的第三个星期五上午10:15触发
每隔5秒执行一次:*/5 * * * * ?
每隔1分钟执行一次:0 */1 * * * ?
每天23点执行一次:0 0 23 * * ?
每天凌晨1点执行一次:0 0 1 * * ?
每月1号凌晨1点执行一次:0 0 1 1 * ?
每月最后一天23点执行一次:0 0 23 L * ?
每周星期天凌晨1点实行一次:0 0 1 ? * L
在26分、29分、33分执行一次:0 26,29,33 * * * ?
每天的0点、13点、18点、21点都执行一次:0 0 0,13,18,21 * * ?

 
 

JAVA精确定时器,利用系统时间,使长期工作的误差稳定。 功能: ·可定时启动任务或直接启动任务 ·重复启动任务(时间间隔可在任务线程中改变,范围大于100ms,否则精度降低) 引用列表: ·import psn.razerpen.thread.AccuracyTimer; ·import psn.razerpen.thread.AccuracyTimerMission; ·import psn.razerpen.time.TimeStruct; 使用方法: //1·继承AccuracyTimerMission接口,创建一个类。 class MyTimer implements AccuracyTimerMission { //2·指定一个周期 int nDelay=1000; //3·重写run方法(如不需要使用新线程执行任务,也可留空) /** * 任务线程,本函数继承自Runnable */ @Override public void run() { System.out.println(new TimeStruct()); } //4·重写RunInCurrentThread(long nCurrentMilliSecond)方法。该方法接收当前时间,并返回下一次执行的时间。如果返回值不大于nCurrentMilliSecond则中止计时器。该方法必须重写。 /** * 接收当前时间的毫秒值,并返回下一次执行的毫秒值。如果返回的下一个时间早于当前时间,则退出 */ @Override public long RunInCurrentThread(long nCurrentMilliSecond) { return nCurrentMilliSecond+=nDelay; } } //5·创建主线程代码 public class TestTimer { public static void main(String[] args) throws InterruptedException { //6·创建一个AccuracyTimer对象,并指定一个任务。 AccuracyTimer at=new AccuracyTimer(new MyTimer()); //7·(可选)如果不需要在新线程中启动任务,则写 // at.SetNewThreadEnabled(false); //否则不写或者写 // at.SetNewThreadEnabled(true); //8·(可选)设定第一次启动的时间点SetNextMissionTime/SetNextMissionMilliSecond或延迟时间SetNextMissionMilliSecondFromNow //设置为当前这一分钟的第59秒后启动(不写此行则表示直接启动) at.SetNextMissionTime(Integer.MIN_VALUE, -1, -1, -1, -1, 59, 0); //9·启动定时器 at.Start(); //10·主线程继续 for(int i=0;i<60;++i){ Thread.sleep(1000); } //11·结束定时器 at.End(); } } 详见sample.razerpen.thread包中TestTimer及各代码文件中注释
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值