วิธีการตั้งค่าตัวจับเวลาใน Java


151

วิธีการตั้งค่าตัวจับเวลาพูด 2 นาทีเพื่อลองเชื่อมต่อกับฐานข้อมูลแล้วโยนข้อยกเว้นหากมีปัญหาใด ๆ ในการเชื่อมต่อ?


1
Coul OP ชี้แจงหากพวกเขาต้องการที่จะพยายามดำเนินการอย่างง่ายเป็นเวลาอย่างน้อย 2 นาทีหรือถ้าข้อยกเว้นจะต้องถูกโยนทิ้งในภายหลังในอีกสองนาทีแม้ว่าความพยายามในการเชื่อมต่อกำลังดำเนินการอยู่ในปัจจุบัน
thecoshman

คำตอบ:


280

ดังนั้นส่วนแรกของคำตอบคือทำอย่างไรในสิ่งที่ผู้ถามถามเพราะนี่คือวิธีที่ฉันตีความตอนแรกและบางคนดูเหมือนว่าจะเป็นประโยชน์ คำถามนั้นได้รับการชี้แจงและฉันได้ขยายคำตอบไปยังที่อยู่นั้น

การตั้งเวลา

ก่อนอื่นคุณต้องสร้างตัวจับเวลา (ฉันใช้java.utilรุ่นที่นี่):

import java.util.Timer;

..

Timer timer = new Timer();

เมื่อต้องการเรียกใช้งานทันทีที่คุณทำ:

timer.schedule(new TimerTask() {
  @Override
  public void run() {
    // Your database code here
  }
}, 2*60*1000);
// Since Java-8
timer.schedule(() -> /* your database code here */, 2*60*1000);

หากต้องการให้งานซ้ำหลังจากระยะเวลาที่คุณต้องทำ:

timer.scheduleAtFixedRate(new TimerTask() {
  @Override
  public void run() {
    // Your database code here
  }
}, 2*60*1000, 2*60*1000);

// Since Java-8
timer.scheduleAtFixedRate(() -> /* your database code here */, 2*60*1000, 2*60*1000);

ทำให้หมดเวลางาน

ในการทำสิ่งที่คำถามที่ได้รับการชี้แจงโดยเฉพาะซึ่งกำลังพยายามทำงานตามระยะเวลาที่กำหนดคุณสามารถทำสิ่งต่อไปนี้:

ExecutorService service = Executors.newSingleThreadExecutor();

try {
    Runnable r = new Runnable() {
        @Override
        public void run() {
            // Database task
        }
    };

    Future<?> f = service.submit(r);

    f.get(2, TimeUnit.MINUTES);     // attempt the task for two minutes
}
catch (final InterruptedException e) {
    // The thread was interrupted during sleep, wait or join
}
catch (final TimeoutException e) {
    // Took too long!
}
catch (final ExecutionException e) {
    // An exception from within the Runnable task
}
finally {
    service.shutdown();
}

สิ่งนี้จะดำเนินการตามปกติโดยมีข้อยกเว้นหากงานเสร็จภายใน 2 นาที หากมันนานกว่านั้น TimeoutException จะถูกโยนทิ้ง

ปัญหาหนึ่งคือแม้ว่าคุณจะได้รับ TimeoutException หลังจากผ่านไปสองนาทีงานจะยังคงทำงานต่อไปแม้ว่าในที่สุดการเชื่อมต่อฐานข้อมูลหรือเครือข่ายจะหมดเวลาและอาจเกิดข้อยกเว้นในเธรด แต่ระวังให้ดีอาจใช้ทรัพยากรจนกว่าจะเกิดขึ้น


โอ้ฉันคิดว่าคุณมีปัญหาของฉันผิดฉันต้องพยายามซ้ำ ๆ เพื่อเชื่อมต่อกับฐานข้อมูลจนถึง 2 นาทีทุกครั้ง
Ankita

ฉันขอโทษ แต่ความต้องการของฉันคือฉันต้องการให้ 2 นาทีเพื่อเชื่อมต่อกับฐานข้อมูลแล้วมันก็จะโยนข้อผิดพลาดบางอย่างมันเหมือนฉันต้องการที่จะกำหนดเวลาที่จะพยายามเชื่อมต่อกับฐานข้อมูล
Ankita

7
ผู้อ่านระวัง: คำตอบนี้ผิดอย่างโจ๋งครึ่ม มันตั้งค่าตัวจับเวลาเพื่อรอ 2 นาทีแล้วเรียกใช้แบบสอบถาม DB หลังจากหน่วงเวลา 2 นาทีนั้นและจากนั้นจะเรียกใช้อีกครั้งและอีกครั้งทุก 2 นาที สิ่งที่ไม่ทำคือการเรียกใช้แบบสอบถาม DB ทันทีและล้มเหลวหลังจาก 2 นาทีซึ่งเป็นสิ่งที่ฉันคิดว่าคำถามที่ถาม (ตามที่อธิบายไว้ในความคิดเห็น)
AgilePro

2
@AgilePro นั่นเป็นเรื่องจริง ฉันไม่ได้ตอบคำถามตามที่ระบุไว้ในที่สุดและฉันได้อัปเดตเป็นที่อยู่นั้นแล้ว
andrewmu

1
@ErnestasGruodis API หลักแสดงรายการตัวสร้างเป็นสาธารณะ: docs.oracle.com/javase/7/docs/api/java/util/Timer.html#Timer ()คุณอาจมีคลาสตัวจับเวลาอื่นใน classpath ของคุณ - ลองใช้ java util.Timer เป็นคลาส
andrewmu

25

ใช้สิ่งนี้

long startTime = System.currentTimeMillis();
long elapsedTime = 0L.

while (elapsedTime < 2*60*1000) {
    //perform db poll/check
    elapsedTime = (new Date()).getTime() - startTime;
}

//Throw your exception

1
อืม ... มันใช้งานได้ในทางเทคนิคยกเว้นกรณีที่ไม่ได้ครอบคลุมกรณีที่คุณหมดเวลาตามที่คุณกำลังดำเนินการสำรวจความคิดเห็นฐานข้อมูลซึ่ง OP อาจจะต้องใช้
thecoshman

1
นี่เป็นคำตอบเดียวที่ถูกต้อง มันจะทำงานในหัวข้อเดียวและจะคำนวณเวลาสิ้นสุดโดยไม่ต้องดริฟท์ การใช้ TimerTask เป็นสิ่งที่ผิดในกรณีนี้ ฉันประหลาดใจที่มีตัวอย่างจำนวนมากใน StackOverflow ที่แนะนำสิ่งผิดประเภทเดียวกันนี้
AgilePro

1
@thecoshman - การใช้งานตัวจับเวลาไม่ขัดจังหวะการทำงานของ DB เมื่อมันกำลังดำเนินการอยู่และไม่ทำเช่นนี้ ไม่ว่าในกรณีใดคุณสามารถควบคุมเวลาที่คุณเริ่มการทำงานของฐานข้อมูลได้เท่านั้น มีความเข้าใจผิดร่วมกันที่คุณต้องมี "ตัวจับเวลา" เพื่อทำสิ่งต่าง ๆ แต่คุณไม่ต้องการ คุณต้องทำอะไรบางอย่างและโดยใช้เวลาปัจจุบันให้แน่ใจว่าคุณไม่ได้ทำมันนานเกินไป
AgilePro

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

14
นี่เป็นสไตล์การเขียนโค้ดที่แย่มากเนื่องจากในขณะที่ลูปทำงานอย่างต่อเนื่องและตรวจสอบสิ่งต่าง ๆ ... ไม่ดีสำหรับซีพียูและไม่ดีสำหรับอายุการใช้งานแบตเตอรี่
ไม่มีที่สิ้นสุด

11

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

เช่น:

FutureTask<Void> task = new FutureTask<Void>(new Callable<Void>() {
  @Override
  public Void call() throws Exception {
    // Do DB stuff
    return null;
  }
});

Executor executor = Executors.newSingleThreadScheduledExecutor();
executor.execute(task);

try {
  task.get(5, TimeUnit.SECONDS);
}
catch(Exception ex) {
  // Handle your exception
}

มีข้อผิดพลาดเกิดขึ้น แต่โปรแกรมไม่ถูกยกเลิกโปรดช่วยฉันผ่านสิ่งนี้
Ankita

2
คุณสามารถใช้คลาส ExecutorService แทน Executor มันมีวิธีการปิด () เพื่อหยุดการปฏิบัติการ
พิธีกรรม

1
ผู้บริหารจะกลืนข้อยกเว้นที่คุณไม่ได้จัดการกับ Callable / Runnable โดยเฉพาะ หาก Runnable / Callable ของคุณไม่สามารถตรวจจับและจัดการข้อยกเว้นได้และคุณได้รับมันจากนั้นคุณจะต้อง subclass ScheduledExecutorService และแทนที่ afterExecute อาร์กิวเมนต์ที่สองเป็น afterecute จะเป็นแบบโยนได้จาก Runnable / Callable
adam

3
    new java.util.Timer().schedule(new TimerTask(){
        @Override
        public void run() {
            System.out.println("Executed...");
           //your code here 
           //1000*5=5000 mlsec. i.e. 5 seconds. u can change accordngly 
        }
    },1000*5,1000*5); 

เพียงคัดลอกรหัสข้างต้นศัตรูพืชมันจะทำงานได้ดี ฉันได้ให้ 'เวลา' สองครั้งแรกเป็นครั้งแรกในการเรียกใช้รหัสนี้และที่สองเป็นช่วงเวลา
Mahadev Mane

เอกสารสำหรับตัวจับเวลาแนะนำให้ใช้กรอบงาน Executor แทน docs.oracle.com/javase/10/docs/api/java/util/Timer.html
Karan Khanna

1

[Android] ถ้ามีคนมองในการดำเนินการจับเวลาเกี่ยวกับหุ่นยนต์โดยใช้ Java

คุณต้องใช้ เธรด UIเช่นนี้เพื่อดำเนินการ

Timer timer = new Timer();
timer.schedule(new TimerTask() {
           @Override
            public void run() {
                ActivityName.this.runOnUiThread(new Runnable(){
                    @Override
                      public void run() {
                       // do something
                      }        
                });
            }
        }, 2000));
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.