Java 每月计时器

原文 标签 java multithreading timer

我正在尝试创建一个将在每个月的同一天运行的 Timer/TimerTask。我无法安排重复的计时器,因为一个月的时间并不总是相同的。

所以,这是我的解决方案:

public class MyTask extends TimerTask {
    public void run(){
        //do process file stuff

        if(scheduledExecutionTime() != 0){
            TimerHelper.restartMyTimer();
        }
    }
}

public class TimerHelper {
    public static HashTable timersTable = new HashTable();

    public static void restartMyTimer(){
        Calendar runDate = Calendar.getInstance();
        runDate.set(Calendar.DAY_OF_MONTH, 1);
        runDate.set(Calendar.HOUR_OF_DAY, 4);
        runDate.set(Calendar.MINUTE, 0);
        runDate.add(Calendar.MONTH, 1);//set to next month

        MyTask myTask = new MyTask();
        Timer myTimer = new Timer();

        myTimer.schedule(myTask, runDate.getTime());

        timersTable = new HashTable();//keeping a reference to the timer so we 
        timersTable.put("1", myTimer);//have the option to cancel it later
    }
}

我想我会遇到的问题是,因为第一个 TimerTask 创建了第二个 Timer,第一个 Timer 是否会因为它创建了第二个而保留下来?代码在第一个 Timer 上完成后,垃圾收集会处理该线程和对象吗?随着时间的推移,我不想建立一堆没有做任何事情但没有被删除的线程。也许我对线程和计时器的工作方式没有正确的理解......

只要我不必使用第三方 JAR,我愿意接受其他创建每月计时器的方法的建议。

谢谢!

最佳答案

如果您担心创建不需要的对象,您总是可以创建一个对象,该对象反过来创建/“销毁”所有引用,因此创建的对象可能会被 gc'ed。

在最坏的情况下,你一年会有 12 个不需要的物品,我认为这是可以忍受的。你的担心还是有道理的。

这是我在执行结束时遵循乔尔建议的时间表的尝试。注意,当前的 Timer 被一个新的 Timer 替换,因此,计时器和计时器任务都可以被 gc'ed。

package monthly.schedule;

import java.util.Timer;
import java.util.TimerTask;
import java.util.Date;
import java.util.Calendar;

public class MonthlyTimer { 
    // What to do
    private final Runnable whatToDo;

    // when 
    private final int dayOfMonth;
    private final int hourOfDay;

    // The current timer
    private Timer current = new Timer();//to avoid NPE

    public void cancelCurrent() { 
        current.cancel();// cancel this execution;
        current.purge(); // removes the timertask so it can be gc'ed
    }

    // create a new instance
    public static MonthlyTimer schedule( Runnable runnable, int dayOfMonth, int hourOfDay ) { 
        return new MonthlyTimer( runnable, dayOfMonth, hourOfDay );
    }

    private MonthlyTimer(Runnable runnable, int day, int hour ) { 
        this.whatToDo = runnable;
        this.dayOfMonth = day;
        this.hourOfDay = hour;
        schedule();
    }
    // Schedules the task for execution on next month. 
    private void schedule() { 
        // Do you mean like this?
        cancelCurrent();
        current = new Timer(); // assigning a new instance
        // will allow the previous Timer to be gc'ed

        current.schedule( new TimerTask() { 
            public void run() { 
                try { 
                    whatToDo.run();
                } finally { 
                    schedule();// schedule for the next month
                }
            }
        } , nextDate() );           
    }
    // Do the next date stuff
    private Date nextDate() { 
        Calendar runDate = Calendar.getInstance();
        runDate.set(Calendar.DAY_OF_MONTH, dayOfMonth);
        runDate.set(Calendar.HOUR_OF_DAY, hourOfDay);
        runDate.set(Calendar.MINUTE, 0);
        runDate.add(Calendar.MONTH, 1);//set to next month
        return runDate.getTime();
    }
}

class UseIt { 
    public static void main( String [] args ) { 
        int the1st = 1;
        int at16hrs = 16;

        MonthlyTimer t = MonthlyTimer.schedule( new Runnable() { 
            public void run() { 
                System.out.println( "Hola" );
            }}, the1st, at16hrs );

        // will print "Hola" every 1st at 16:00 hrs.
       // if needed you can cancel with: 
        t.cancelCurrent();

    }
}

关于Java 每月计时器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4107011/

相关文章:

java - 调用数据库时应用程序卡住

c# - 线程和 sleep ()

c# - 控制台应用程序中的倒数计时器

javascript - Android/phonegap - 点击响应时间慢

java - 如何调试大型的面向插件的Java项目,例如Eclipse中的ActiveMQ(非Artemis)?

java - 使用Jedis在单个管道中进行多个Redis事务(多个/exec)

java - JavaFX:运行并行线程并将整体进度更新为ProgressBar

swift - 每天午夜运行代码

java - 订单检查条件

java - 从命令行 java 程序访问 php session