जावा: सेकंड की एक विशिष्ट संख्या के बाद एक फ़ंक्शन चलाते हैं


148

मेरा एक विशिष्ट कार्य है जिसे मैं 5 सेकंड के बाद निष्पादित करना चाहता हूं। मैं जावा में कैसे कर सकता हूं?

मुझे javax.swing.timer मिला, लेकिन मैं वास्तव में समझ नहीं पा रहा हूं कि इसका उपयोग कैसे किया जाए। ऐसा लगता है कि मैं किसी तरह से सरल की तलाश कर रहा हूं तो यह वर्ग प्रदान करता है।

कृपया एक सरल उपयोग उदाहरण जोड़ें।


क्या आप 5 सेकंड इंतजार करना चाहते हैं और फिर कुछ निष्पादित करते हैं या क्या आप 5 सेकंड में कुछ और करना जारी रखना चाहते हैं?
व्हिस्कीसियर

मैं कुछ और करना जारी रखना चाहता हूँ
ufk

जवाबों:


229
new java.util.Timer().schedule( 
        new java.util.TimerTask() {
            @Override
            public void run() {
                // your code here
            }
        }, 
        5000 
);

संपादित करें:

जावदोक कहते हैं:

एक टाइमर ऑब्जेक्ट के अंतिम लाइव संदर्भ के बाद चला जाता है और सभी उत्कृष्ट कार्यों ने निष्पादन को पूरा कर लिया है, टाइमर का कार्य निष्पादन धागा सुशोभित होता है (और कचरा संग्रह के अधीन हो जाता है)। हालांकि, ऐसा होने में मनमाना समय लग सकता है।


1
यदि आप वह कोड चलाते हैं, तो आप थ्रेड लीक करेंगे। समाप्त होने पर टाइमर को साफ करना सुनिश्चित करें।
स्कैफ़मैन

1
@skaffman: मैंने javadoc से एक बयान जोड़ा। क्या आपको कॉल शेड्यूल के बाद वास्तव में सफाई करनी है?
tangens

1
यह ठीक हो सकता है, लेकिन तब यह नहीं हो सकता है। यदि आप उस कोड के टुकड़े को कई बार चलाते हैं, तो आपके पास ढीले धागे होंगे, जिनके बारे में कोई मतलब नहीं होगा।
स्केफमैन

5
import java.util.Timer; import java.util.TimerTask;यह अधिक स्पष्ट हो सकता है कि यह नहीं है javax.swing.Timer। / ध्यान दें, यदि आप स्विंग (और वास्तव में AWT) का उपयोग कर रहे हैं, तो आपको गैर-ईवेंट डिस्पैच थ्रेड (EDT) थ्रेड्स ( java.util.Timerकार्य खराब; javax.swing.Timerक्रिया अच्छी) पर घटकों को बदलने के लिए कुछ भी नहीं करना चाहिए ।
टॉम हॉकिन -

2
@PaAAlexander डॉक्स के अनुसार - cancelरन विधि के अंत में टाइमर की विधि को कॉल करने से टाइमर के निष्पादन थ्रेड स्पष्ट हो जाएंगे।
डंडालफ

58

कुछ इस तरह:

// When your program starts up
ScheduledExecutorService executor = Executors.newSingleThreadScheduledExecutor();

// then, when you want to schedule a task
Runnable task = ....    
executor.schedule(task, 5, TimeUnit.SECONDS);

// and finally, when your program wants to exit
executor.shutdown();

विभिन्न अन्य कारखाने विधियां हैं, Executorजिन पर आप इसके बजाय उपयोग कर सकते हैं, यदि आप पूल में अधिक धागे चाहते हैं।

और याद रखें, जब आप काम पूरा कर लें, तो निष्पादक को बंद करना महत्वपूर्ण है। shutdown()विधि सफाई से थ्रेड पूल बंद हो जाएगा जब अंतिम कार्य पूरा कर लिया है, और जब तक ऐसा होता है रोकेंगे। shutdownNow()थ्रेड पूल को तुरंत समाप्त कर देगा।


24

उपयोग करने का उदाहरण javax.swing.Timer

Timer timer = new Timer(3000, new ActionListener() {
  @Override
  public void actionPerformed(ActionEvent arg0) {
    // Code to be executed
  }
});
timer.setRepeats(false); // Only execute once
timer.start(); // Go go go!

यह कोड केवल एक बार निष्पादित किया जाएगा, और निष्पादन 3000 एमएस (3 सेकंड) में होता है।

जैसा कि कैमिक्र का उल्लेख है, आपको एक संक्षिप्त परिचय के लिए " हाउ टू यूज स्विंग टाइमर " का पता लगाना चाहिए ।


6

मेरा कोड इस प्रकार है:

new java.util.Timer().schedule(

    new java.util.TimerTask() {
        @Override
        public void run() {
            // your code here, and if you have to refresh UI put this code: 
           runOnUiThread(new   Runnable() {
                  public void run() {
                            //your code

                        }
                   });
        }
    }, 
    5000 
);

5

@Tangens उत्तर की भिन्नता के रूप में: यदि आप कचरा कलेक्टर को अपने धागे को साफ करने के लिए इंतजार नहीं कर सकते हैं, तो अपने रन विधि के अंत में टाइमर को रद्द करें।

Timer t = new java.util.Timer();
t.schedule( 
        new java.util.TimerTask() {
            @Override
            public void run() {
                // your code here
                // close the thread
                t.cancel();
            }
        }, 
        5000 
);

Timer tघोषित नहीं किया जाना चाहिए finalक्योंकि यह एक आंतरिक वर्ग के अंदर पहुँचा जा रहा है?
जोशुआ पिंटर

1
@JoshuaPinter हाँ, इसे अंतिम घोषित किया जाना चाहिए लेकिन इसे कम से कम जावा 8 में स्पष्ट रूप से अंतिम रूप से घोषित करने की आवश्यकता नहीं है। इसे "प्रभावी रूप से अंतिम" ( javarevisited.blogspot.com/2015/03/ ))
Dandalf

4

आपके मूल प्रश्न में "स्विंग टाइमर" का उल्लेख है। यदि वास्तव में आपका सवाल स्वांग से संबंधित है, तो आपको स्विंग टाइमर का उपयोग करना चाहिए और उपयोग नहीं करना चाहिए।

अधिक जानकारी के लिए " कैसे टाइमर का उपयोग करें " पर स्विंग ट्यूटोरियल से अनुभाग पढ़ें ।


4

आप Thread.Sleep () फ़ंक्शन का उपयोग कर सकते हैं

Thread.sleep(4000);
myfunction();

आपका कार्य 4 सेकंड के बाद निष्पादित होगा। हालाँकि यह पूरे कार्यक्रम को विराम दे सकता है ...


और यह केवल गारंटी देता है कि निष्पादन 4sec के बाद चलेगा, जिसका मतलब 10 सेकंड के बाद भी हो सकता है!
क्वेस्टजन

2
searchzen, आप पाएंगे कि यहाँ सभी विधियाँ ऐसा करती हैं। वास्तव में, भले ही आप ओएस स्तर पर कुछ निर्धारित कर रहे हों, आप आम तौर पर एक घटना से पहले केवल न्यूनतम बीता हुआ समय की गारंटी दे सकते हैं।
एथन

यह वास्तविक प्रश्न नहीं है
इंदर आर सिंह

मुझे बस इसे एक डाउनवोट देना था - प्रश्न के उत्तर पर बिल्कुल भी नहीं।
TheMayer

ओपी ने एक टिप्पणी में कहा "मैं कुछ और करना जारी रखना चाहता हूं"; जाहिर है, यह कोड नहीं करता है।
अभिजीत सरकार

3

ScheduledThreadPoolExecutor यह क्षमता है, लेकिन यह काफी भारी है।

Timer यह भी क्षमता है लेकिन केवल एक बार उपयोग किए जाने पर भी कई धागे खोलता है।

यहां एक परीक्षण (Android के हैंडलर के पास हस्ताक्षर के साथ एक सरल कार्यान्वयन है। पोस्टडायलेड () ):

public class JavaUtil {
    public static void postDelayed(final Runnable runnable, final long delayMillis) {
        final long requested = System.currentTimeMillis();
        new Thread(new Runnable() {
            @Override
            public void run() {
                // The while is just to ignore interruption.
                while (true) {
                    try {
                        long leftToSleep = requested + delayMillis - System.currentTimeMillis();
                        if (leftToSleep > 0) {
                            Thread.sleep(leftToSleep);
                        }
                        break;
                    } catch (InterruptedException ignored) {
                    }
                }
                runnable.run();
            }
        }).start();
    }
}

परीक्षा:

@Test
public void testRunsOnlyOnce() throws InterruptedException {
    long delay = 100;
    int num = 0;
    final AtomicInteger numAtomic = new AtomicInteger(num);
    JavaUtil.postDelayed(new Runnable() {
        @Override
        public void run() {
            numAtomic.incrementAndGet();
        }
    }, delay);
    Assert.assertEquals(num, numAtomic.get());
    Thread.sleep(delay + 10);
    Assert.assertEquals(num + 1, numAtomic.get());
    Thread.sleep(delay * 2);
    Assert.assertEquals(num + 1, numAtomic.get());
}

एक पाश में बुलाया चेतावनी नींद देता है
शरीफ

whileबस रुकावट अनदेखी करने के लिए है।
एलिकएल्ज़िन-किलाका

2

अन्य सभी unswers को आपके कोड को एक नए थ्रेड के अंदर चलाने की आवश्यकता होती है। कुछ सरल उपयोग मामलों में आप बस थोड़ा इंतजार करना चाहते हैं और एक ही धागे / प्रवाह के भीतर निष्पादन जारी रख सकते हैं।

नीचे दिया गया कोड उस तकनीक को प्रदर्शित करता है। ध्यान रखें कि यह java.util के समान है। चिमटा हुड के नीचे करता है लेकिन अधिक हल्का होता है।

import java.util.concurrent.TimeUnit;
public class DelaySample {
    public static void main(String[] args) {
       DelayUtil d = new DelayUtil();
       System.out.println("started:"+ new Date());
       d.delay(500);
       System.out.println("half second after:"+ new Date());
       d.delay(1, TimeUnit.MINUTES); 
       System.out.println("1 minute after:"+ new Date());
    }
}

DelayUtil कार्यान्वयन

import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.ReentrantLock;

public class DelayUtil {
    /** 
    *  Delays the current thread execution. 
    *  The thread loses ownership of any monitors. 
    *  Quits immediately if the thread is interrupted
    *  
    * @param duration the time duration in milliseconds
    */
   public void delay(final long durationInMillis) {
      delay(durationInMillis, TimeUnit.MILLISECONDS);
   }

   /** 
    * @param duration the time duration in the given {@code sourceUnit}
    * @param unit
    */
    public void delay(final long duration, final TimeUnit unit) {
        long currentTime = System.currentTimeMillis();
        long deadline = currentTime+unit.toMillis(duration);
        ReentrantLock lock = new ReentrantLock();
        Condition waitCondition = lock.newCondition();

        while ((deadline-currentTime)>0) {
            try {
                lock.lockInterruptibly();    
                waitCondition.await(deadline-currentTime, TimeUnit.MILLISECONDS);
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
                return;
            } finally {
                lock.unlock();
            }
            currentTime = System.currentTimeMillis();
        }
    }
}

2
public static Timer t;

public synchronized void startPollingTimer() {
        if (t == null) {
            TimerTask task = new TimerTask() {
                @Override
                public void run() {
                   //Do your work
                }
            };

            t = new Timer();
            t.scheduleAtFixedRate(task, 0, 1000);
        }
    }

2
हालांकि यह कोड प्रश्न का उत्तर दे सकता है, लेकिन यह कोड क्यों और / या इस प्रश्न के उत्तर के बारे में अतिरिक्त संदर्भ प्रदान करता है, इसके दीर्घकालिक मूल्य में सुधार करता है।
मटुस
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.