8

我正在开发计算机视觉应用程序,我将需要分类器类。此类在每次运行应用程序时都是不可变的,它会在初始化时从磁盘加载训练数据。我想确保整个程序都可以访问相同的训练数据,并且我想在加载它们后阻止从磁盘重新加载。

我正在考虑的是使用静态类或单例。我不确定如何将数据加载到静态类,因为在编译时不知道数据文件的路径 - 这将是程序参数。所以我在考虑单例模式,但我不知道如何动态初始化它。

我的想法是使用以下内容:

class Singleton {
    private static Singleton instance;
    private Singleton() { ... }
    private static SomeDataObject data;

    public static Singleton getInstance() {
        if(instance == null)
            instance = new Singleton();

            return instance;
    }

    public static init(string dataPath){
        if(data == null)
             loadDataFromFile(dataPath)
    }
}

这是行不通的,因为我无法控制首先调用哪个方法。

我知道正确的方法是在开始时使用数据创建实例并将其传递给所有需要它的类和方法,但这并不是真正的通用解决方案。我可以在自己的代码中跟踪对分类器的所有调用,但如果我将代码作为 API,这将是一个问题。

总之如何在运行时初始化单例?

4

2 回答 2

8

我不认为(确切地)你想做的事情会奏效。

以下将起作用:

public static void main(String[] args)
{
  Singleton.init("somepath");
  ...
  Singleton.getInstance().doingStuff();
  ...
}

更好的实现可能是:(NullPointerException如果您尝试使用它而不init先调用它会导致 a)(虽然不再是真正的 Singleton)

private static Singleton instance;
private SomeDataObject data;

private Singleton(String path) { loadDataFromFile(path); ... }

public static Singleton getInstance() {
   return instance;
}

public static void init(String dataPath){
   instance = new Singleton(dataPath);
}

然后是:(除了可能的不良编码习惯)

class Main
{
  public static void main(String[] args)
  {
    Singleton.currentPath = "somepath";
    ...
  }
}

class Singleton
{
  public static String currentPath = null;
  private static Singleton instance;
  private SomeDataObject data;

  private Singleton(String path) { loadDataFromFile(path); ... }

  public static Singleton getInstance() {
     if(instance == null && currentPath != null)
        instance = new Singleton(currentPath);
     return instance;
  }
}

我想这并不能真正解决问题。

于 2013-02-19T17:04:31.627 回答
0

我使用的东西比当前获胜的解决方案“更”线程安全,几乎没有使用同步。

import java.util.function.Supplier;

public class InitOnce {

/**
 * Marked as final to prevent JIT reordering
 */
private final Supplier<String> theArgs;

private InitOnce(Supplier<String> supplier) {
    super();
    this.theArgs = supplier;
}

/**
 * Uses the arguments to do something
 * 
 * @return
 */
public String doSomething() {
    return "Something : " + theArgs.get();
}

/**
 * Initializes all the things
 * 
 * @param someArgs
 */
public static synchronized void init(final Supplier<String> someArgs) {
    class InitOnceFactory implements Supplier<InitOnce> {
        private final InitOnce initOnceInstance = new InitOnce(someArgs);

        @Override
        public InitOnce get() {
            return initOnceInstance;
        }
    }

    if (!InitOnceFactory.class.isInstance(instance)) {
        instance = new InitOnceFactory();
    } else {
        throw new IllegalStateException("Already Initialized");
    }
}

private static Supplier<InitOnce> instance = new InitOnceHolder();

/**
 * Temp Placeholder supplier
 * 
 */
private static final class InitOnceHolder implements Supplier<InitOnce> {
    @Override
    public synchronized InitOnce get() {

        if (InitOnceHolder.class.isInstance(instance))
            throw new IllegalStateException("Not Initialized");

        return instance.get();
    }

}

/**
 * Returns the instance
 * 
 * @return
 */
public static final InitOnce getInstance() {
    return instance.get();
}
}
于 2016-08-18T14:40:26.173 回答