Java 依赖于it的测试类';s静态最终字段

Java 依赖于it的测试类';s静态最终字段,java,unit-testing,junit,Java,Unit Testing,Junit,我想知道是否有可能测试依赖于静态final字段的类?例如,我有以下课程: public final class Foo { private static Foo instance; public static final String BAR_FILE = "/usr/bin/bar.bar"; private Foo() { loadBar(); } public static synchronized Foo getInst

我想知道是否有可能测试依赖于静态final字段的类?例如,我有以下课程:

public final class Foo
{
    private static Foo instance;
    public static final String BAR_FILE = "/usr/bin/bar.bar";

    private Foo()
    {
        loadBar();
    }

    public static synchronized Foo getInstance()
    {
        if (instance == null)
        {
            instance = new Foo();
        }
        return instance;
    }

    private void loadBar()
    {
        final Properties prop = new Properties();
        try
        {
            final FileInputStream fis = new FileInputStream(BAR_FILE);
            prop.load(fis);
            fis.close();
        }
        catch (final IOException ex)
        {
            System.out.println("Exception!");
        }
    }
}

所以,如果BAR_文件是硬编码的,并且等于某个unix样式的路径,那么如何在windows上测试
getInstance()
(我知道没有什么需要测试的,但这只是一个示例)方法呢。我试图通过反思来改变这个领域,但没有运气,而且,根据讨论,这甚至是不可能的。感谢您的帮助

将业务逻辑与构造逻辑分开。您的类不应该关心它是如何创建的,将其留给工厂或DI框架

使用DI框架非常简单:

@Singleton
public final class Foo {
  public Foo(String fileName) { loadBar(fileName);}  // Use in tests
  public Foo() { loadBar(DEFAULT_FILENAME);} 
  private void loadBar(String file)
    {
         // SNIP
    }
}


public class ThingieThatUsesFoo {
     public ThingieThatUsesFoo(Foo foo) { ... } // The DI framework (Guice/Spring) will handle the details of WHAT gets injected.
}
使用一个工厂是一个多一点的努力,但不是很多

// You could make FooFactory a singleton if that is your thing.
public class FooFactory {
     private Foo fooInstance = new Foo();
     public Foo getFoo() {
         return fooInstance;
     }
     public void setFooForTestingOnly(Foo fooInstance) { ... } // Set this to a mock or what have thee when testing clients.
}

在这两种情况下,测试Foo的客户端都很容易,因为您可以插入一个模拟,而测试Foo则更容易,因为您现在可以将文件设置为读取(如果您想更聪明,请不要传入文件,而是传入一个读卡器或InputStream)。

在测试方法之前,只需创建一个以静态字段命名的文件。基本上,这个类不是以测试友好的方式设计的。如果你能改变它,避免它成为一个单身汉,那将真的很有帮助。。。你可以很容易地把一个工厂放进去,以避免两次加载同一个文件——如果你真的想的话,这个工厂可以是一个单独的工厂。@SotiriosDelimanolis,不太明白你的意思。@mr.nothing,我想我不明白你的问题是什么。@JonSkeet,我不关心两次读取文件,但是我要多次创建这个对象,这就是为什么我将它设置为单例对象。您能提供一些代码来展示如何使用工厂模式重写它吗?