วิธีการจัดตารางเวลางานเป็นระยะใน Java?


183

ฉันต้องการกำหนดเวลางานให้ทำงานในช่วงเวลาที่แน่นอน ฉันจะทำสิ่งนี้ได้อย่างไรด้วยการสนับสนุนช่วงเวลาที่ยาวนาน (ตัวอย่างเช่นในแต่ละ 8 ชั่วโมง)

java.util.Timer.scheduleAtFixedRateฉันกำลังใช้ ไม่java.util.Timer.scheduleAtFixedRateสนับสนุนช่วงเวลาที่ยาว?

คำตอบ:


260

ใช้ScheduledExecutorService :

 private final ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(1);
 scheduler.scheduleAtFixedRate(yourRunnable, 8, 8, TimeUnit.HOURS);

1
หากคุณต้องการนี้ทำงานทุกวันในเวลาที่กำหนดมีไม่เป็นวิธีที่ดีที่จะทำมันเพราะTimeUnitนำไปใช้ทั้งในและinitialDelay periodทำงานทุก 24 ชั่วโมงจะจบลงด้วยการถูกโยนออกเมื่อเตะเวลาใน แต่TimeUnitของไม่ยอมให้คุณระบุเม็ดเล็กDAYS initialDelay(ฉันคิดว่าการใช้งาน ScheduledExecutorService ภายในจะแปลงDAYSเป็นนาโนวินาทีอยู่ดี)
Sam Barnum

46

คุณควรดูQuartzว่าเป็น Java framework ที่ทำงานกับรุ่น EE และ SE และอนุญาตให้กำหนดงานเพื่อดำเนินการตามเวลาที่กำหนด


23

ลองใช้วิธีนี้ ->

ขั้นแรกสร้างคลาส TimeTask ที่รันงานของคุณดูเหมือน:

public class CustomTask extends TimerTask  {

   public CustomTask(){

     //Constructor

   }

   public void run() {
       try {

         // Your task process

       } catch (Exception ex) {
           System.out.println("error running thread " + ex.getMessage());
       }
    }
}

จากนั้นในคลาสหลักคุณสร้างอินสแตนซ์ของภารกิจและรันเป็นระยะ ๆ ตามวันที่ที่ระบุ:

 public void runTask() {

        Calendar calendar = Calendar.getInstance();
        calendar.set(
           Calendar.DAY_OF_WEEK,
           Calendar.MONDAY
        );
        calendar.set(Calendar.HOUR_OF_DAY, 15);
        calendar.set(Calendar.MINUTE, 40);
        calendar.set(Calendar.SECOND, 0);
        calendar.set(Calendar.MILLISECOND, 0);



        Timer time = new Timer(); // Instantiate Timer Object

        // Start running the task on Monday at 15:40:00, period is set to 8 hours
        // if you want to run the task immediately, set the 2nd parameter to 0
        time.schedule(new CustomTask(), calendar.getTime(), TimeUnit.HOURS.toMillis(8));
}

6
เพื่อให้โค้ดอ่านง่ายขึ้นคุณสามารถเปลี่ยนอาร์กิวเมนต์สุดท้ายในกำหนดเวลาการโทรไปยัง TimeUnit.HOURS.toMillis (8)
darrenmc

เอกสารสำหรับตัวจับเวลาแนะนำให้ใช้กรอบงาน Executor แทน
Karan Khanna

14

ใช้ Google Guava AbstractScheduledServiceตามที่ระบุด้านล่าง:

public class ScheduledExecutor extends AbstractScheduledService
{
   @Override
   protected void runOneIteration() throws Exception
   {
      System.out.println("Executing....");
   }

   @Override
   protected Scheduler scheduler()
   {
        return Scheduler.newFixedRateSchedule(0, 3, TimeUnit.SECONDS);
   }

   @Override
   protected void startUp()
   {
       System.out.println("StartUp Activity....");
   }


   @Override
   protected void shutDown()
   {
       System.out.println("Shutdown Activity...");
   }

   public static void main(String[] args) throws InterruptedException
   {
       ScheduledExecutor se = new ScheduledExecutor();
       se.startAsync();
       Thread.sleep(15000);
       se.stopAsync();
   }

}

หากคุณมีบริการเพิ่มเติมเช่นนี้การลงทะเบียนบริการทั้งหมดใน ServiceManager จะดีเพราะบริการทั้งหมดสามารถเริ่มและหยุดพร้อมกันได้ อ่านที่นี่สำหรับข้อมูลเพิ่มเติมเกี่ยวกับ ServiceManager


9

หากคุณต้องการที่จะยึดติดกับjava.util.Timerคุณสามารถใช้มันเพื่อกำหนดเวลาในช่วงเวลาที่มีขนาดใหญ่ คุณแค่ผ่านช่วงเวลาที่คุณกำลังถ่ายทำ ตรวจสอบเอกสารที่นี่


5

สองคลาสเหล่านี้สามารถทำงานร่วมกันเพื่อกำหนดตารางงานเป็นระยะ:

งานที่กำหนดไว้

import java.util.TimerTask;
import java.util.Date;

// Create a class extending TimerTask
public class ScheduledTask extends TimerTask {
    Date now; 
    public void run() {
        // Write code here that you want to execute periodically.
        now = new Date();                      // initialize date
        System.out.println("Time is :" + now); // Display current time
    }
}

เรียกใช้งานที่กำหนดเวลาไว้

import java.util.Timer;

public class SchedulerMain {
    public static void main(String args[]) throws InterruptedException {
        Timer time = new Timer();               // Instantiate Timer Object
        ScheduledTask st = new ScheduledTask(); // Instantiate SheduledTask class
        time.schedule(st, 0, 1000);             // Create task repeating every 1 sec
        //for demo only.
        for (int i = 0; i <= 5; i++) {
            System.out.println("Execution in Main Thread...." + i);
            Thread.sleep(2000);
            if (i == 5) {
                System.out.println("Application Terminates");
                System.exit(0);
            }
        }
    }
}

ข้อมูลอ้างอิงhttps://www.mkyong.com/java/how-to-run-a-task-periodically-in-java/


ทางออกที่ดีที่สุดจนถึงตอนนี้สะอาดและใช้งานง่ายขึ้น
Salvador Vigo


4

ทำบางสิ่งบางอย่างทุกวินาที

Timer timer = new Timer();
timer.schedule(new TimerTask() {
    @Override
    public void run() {
        //code
    }
}, 0, 1000);

1
เอกสารสำหรับตัวจับเวลาแนะนำให้ใช้เฟรมเวิร์ก Executor แทน
Karan Khanna

3

ฉันใช้คุณสมบัติของ Spring Framework ( jar ฤดูใบไม้ผลิบริบทหรือการพึ่งพา Maven)

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;


@Component
public class ScheduledTaskRunner {

    @Autowired
    @Qualifier("TempFilesCleanerExecution")
    private ScheduledTask tempDataCleanerExecution;

    @Scheduled(fixedDelay = TempFilesCleanerExecution.INTERVAL_TO_RUN_TMP_CLEAN_MS /* 1000 */)
    public void performCleanTempData() {
        tempDataCleanerExecution.execute();
    }

}

ScheduledTaskเป็นอินเทอร์เฟซของฉันเองโดยใช้วิธีการแบบกำหนดเองซึ่งฉันเรียกว่าเป็นงานที่กำหนดเวลาไว้


2

คุณได้ลองSpring Schedulerโดยใช้คำอธิบายประกอบแล้วหรือยัง

@Scheduled(cron = "0 0 0/8 ? * * *")
public void scheduledMethodNoReturnValue(){
    //body can be another method call which returns some value.
}

คุณสามารถทำได้ด้วย xml เช่นกัน

 <task:scheduled-tasks>
   <task:scheduled ref = "reference" method = "methodName" cron = "<cron expression here> -or- ${<cron expression from property files>}"
 <task:scheduled-tasks>

0

servlet ของฉันมีสิ่งนี้เป็นรหัสวิธีเก็บไว้ในตัวกำหนดตารางเวลาหากผู้ใช้กดยอมรับ

if(bt.equals("accept")) {
    ScheduledExecutorService scheduler=Executors.newScheduledThreadPool(1);
    String lat=request.getParameter("latlocation");
    String lng=request.getParameter("lnglocation");
    requestingclass.updatelocation(lat,lng);
}
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.