Path: csiph.com!x330-a1.tempe.blueboxinc.net!usenet.pasdenom.info!news.albasani.net!.POSTED!not-for-mail From: Jan Burse Newsgroups: comp.lang.java.programmer Subject: Re: Lightweight postDelayed / removeCallbacks Date: Mon, 06 Feb 2012 01:15:14 +0100 Organization: albasani.net Lines: 136 Message-ID: References: Mime-Version: 1.0 Content-Type: text/plain; charset=ISO-8859-15; format=flowed Content-Transfer-Encoding: 7bit X-Trace: news.albasani.net cZmzHNnhssYVDYZW5bITk75I8nxQe5MrJYb6kRCw1TF4owz69t1CuAlQ6vIbz7Yu92zsonLwjuvbIdNvvmjJ3g== NNTP-Posting-Date: Mon, 6 Feb 2012 00:15:15 +0000 (UTC) Injection-Info: news.albasani.net; logging-data="lKZHlZbau5uVBVQy+mjZA9wvKc6gH5ZKgOX/ZTJBE25w0MwBLbaP18fKiEOPEeN3Hz0UpSqYJqGUFQcu2iNqSPA19Lk/VxunZX4mpzIcO9C908+7MwD+tSnsMtIjxYx+"; mail-complaints-to="abuse@albasani.net" User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10.6; rv:10.0) Gecko/20120129 Firefox/10.0 SeaMonkey/2.7 In-Reply-To: Cancel-Lock: sha1:UEiPMwsRnLx7A8m6DxiVkoxAM74= Xref: x330-a1.tempe.blueboxinc.net comp.lang.java.programmer:11751 Steven Simpson schrieb: > A first stab: Thank you very much for your solution. Since I was in need of a solution I implemented already something yesterday. Difference to your stab: postDelayed() and removeCallbacks() do not wait for mouse / keyevents / etc.. to flush. The problem is that invokeAndWait() calls postEvent(), which calls flushPendingEvents() in turn. The Android removeCallbacks() and postDelayed() are also immediate. They call in turn removeMessages() respectively enqueueMessage() which are immediate. I didn't use javax.swing.Timer since I still think they are not leightweight enough. So my waiting is done via Object.wait(), whereas the Swing timer uses Condition.awaitNanos(). Not sure what is more reliable. I also use System.currentTimeMillis() whereas the Swing timer uses System.nanoTime(). nanoTime() is more stable, it does not change when the wall clock is changed on the computer. But current solution is then to use tail recursion for a looping animation. Which can get jerky, since the rate will be not fixed. The delivery of the event might still take some time before a next postDelayed() happens, since it is delivered via postEvent() which uses flushPendingEvents(). But this is tollerated in the current application. Could use a post with runnable = null to stop the thread, something similar is done in the Android looper. But currently the thread has to be started via start(), and can then be easily terminated via interrupt(). ---------------- Begin Code ------------------ public class ThreadTimer extends Thread { private final ArrayList entrylist = new ArrayList(); private final Object entrylock = new Object(); public void postDelayed(Runnable r, int d) { TimerEntry entry = new TimerEntry(r, System.currentTimeMillis() + d); synchronized (entrylock) { for (int i = 0; i < entrylist.size(); i++) { TimerEntry entry2 = entrylist.get(i); if (entry.getWhen() < entry2.getWhen()) { entrylist.add(i, entry); entrylock.notifyAll(); return; } } entrylist.add(entry); entrylock.notifyAll(); } } public void removeCallbacks(Runnable r) { synchronized (entrylock) { int backsize = entrylist.size(); for (int i = entrylist.size() - 1; i >= 0; i--) { TimerEntry entry = entrylist.get(i); if (entry.getRunnable() == r) entrylist.remove(i); } if (backsize != entrylist.size()) entrylock.notifyAll(); } } private TimerEntry take() throws InterruptedException { for (; ; ) { synchronized (entrylock) { if (entrylist.size() == 0) { entrylock.wait(); } else { TimerEntry entry = entrylist.get(0); long now = System.currentTimeMillis(); if (entry.getWhen() <= now) { entrylist.remove(0); return entry; } else { entrylock.wait(entry.getWhen() - now); } } } } } public void run() { try { for (; ; ) { TimerEntry entry = take(); SwingUtilities.invokeLater(entry.getRunnable()); } } catch (InterruptedException x) { /* */ } } } class TimerEntry { private final Runnable runnable; private final long when; TimerEntry(Runnable r, long w) { runnable = r; when = w; } Runnable getRunnable() { return runnable; } long getWhen() { return when; } } ---------------- End Code --------------------