Java 1.4 synchronization: only allow one instance of method to run (non blocking)?

后端 未结 4 969
眼角桃花
眼角桃花 2021-01-20 05:14

I have a class proposing translations utilities. The translations themselves should be reloaded every 30 minutes. I use Spring Timer support for that. Basically, my class lo

相关标签:
4条回答
  • 2021-01-20 05:32

    I am from a .net background(no java experience at all), but you could try a simple static flag of some sort that checks at the beginning of the method if its alrady running. Then all you need to do is make sure any read/write of that flag is synchronized. So at beginning check the flag, if its not set, set it, if it is set, return. If its not set, run the rest of the method, and after its complete, unset it. Just make sure to put the code in a try/finally and the flag iunsetting in the finally so it always gets unset in case of error. Very simplified but may be all you need.

    Edit: This actually probably works better than synchronizing the method. Because do you really need a new translation immediately after the one before it finishes? And you may not want to lock up a thread for too long if it has to wait a while.

    0 讨论(0)
  • 2021-01-20 05:34

    Use some form of locking mechanism to only perform the task if it is not already in progress. Acquiring the locking token must be a one-step process. See:

    /**
     * @author McDowell
     */
    public abstract class NonconcurrentTask implements Runnable {
    
        private boolean token = true;
    
        private synchronized boolean acquire() {
            boolean ret = token;
            token = false;
            return ret;
        }
    
        private synchronized void release() {
            token = true;
        }
    
        public final void run() {
            if (acquire()) {
                try {
                    doTask();
                } finally {
                    release();
                }
            }
        }
    
        protected abstract void doTask();
    
    }
    

    Test code that will throw an exception if the task runs concurrently:

    public class Test {
    
        public static void main(String[] args) {
            final NonconcurrentTask shared = new NonconcurrentTask() {
                private boolean working = false;
    
                protected void doTask() {
                    System.out.println("Working: "
                            + Thread.currentThread().getName());
                    if (working) {
                        throw new IllegalStateException();
                    }
                    working = true;
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException e) {
                        throw new RuntimeException(e);
                    }
                    if (!working) {
                        throw new IllegalStateException();
                    }
                    working = false;
                }
            };
    
            Runnable taskWrapper = new Runnable() {
                public void run() {
                    while (true) {
                        try {
                            Thread.sleep(100);
                        } catch (InterruptedException e) {
                            throw new RuntimeException(e);
                        }
                        shared.run();
                    }
                }
            };
            for (int i = 0; i < 100; i++) {
                new Thread(taskWrapper).start();
            }
        }
    
    }
    
    0 讨论(0)
  • 2021-01-20 05:35

    This is actually identical to the code that is required to manage the construction of a Singleton (gasp!) when done the classical way:

    if (instance == null) {
      synchronized {
        if (instance == null) {
           instance = new SomeClass();
        }
      }
    }
    

    The inner test is identical to the outer test. The outer test is so that we dont routinely enter a synchronised block, the inner test is to confirm that the situation has not changed since we last made the test (the thread could have been preempted before entering Synchronized).

    In your case:

    if (translationsNeedLoading()) {
      synchronized {
        if (translationsNeedLoading()) {
           loadTranslations();
        }
      }
    }
    

    UPDATE: This way of constructing a singleton will not work reliably under your JDK1.4. For explanation see here. However I think you are you will be OK in this scenario.

    0 讨论(0)
  • 2021-01-20 05:54

    Keep a handle on the load thread to see if it's running?

    Or can't you just use a synchronized flag to indicate if a load is in progress?

    0 讨论(0)
提交回复
热议问题