48

我正在开发使用 Android 的 java.xml.parsers.DocumentBuilder 和 DocumentBuilderFactory 实现从 XML 文件加载信息的软件。我正在为我的对象编写单元测试,我需要能够提供各种 xml 文件来执行被测代码。我正在使用 Eclipse 并且有一个单独的 Android 测试项目。我找不到将测试 xml 放入测试项目的方法,以便被测代码可以打开文件。

  • 如果我将文件放在测试项目的 /assets 中,被测代码看不到它。
  • 如果我将文件放在被测代码的 /assets 中,它当然可以看到这些文件,但现在我的实际系统中只包含测试数据文件。
  • 如果我将文件手动复制到 /sdcard/data 目录,我可以从被测代码中打开它们,但这会干扰我的测试自动化。

任何关于如何让不同的 xml 测试文件驻留在测试包中但对被测代码可见的建议将不胜感激。

这是我尝试构建单元测试的方式:

public class AppDescLoaderTest extends AndroidTestCase
{
  private static final String SAMPLE_XML = "sample.xml";

  private AppDescLoader       m_appDescLoader;
  private Application         m_app;

  protected void setUp() throws Exception
  {
    super.setUp();
    m_app = new Application();
    //call to system under test to load m_app using
    //a sample xml file
    m_appDescLoader = new AppDescLoader(m_app, SAMPLE_XML, getContext());
  }

  public void testLoad_ShouldPopulateDocument() throws Exception
  {
    m_appDescLoader.load();

  }    
}

这不起作用,因为 SAMPLE_XML 文件在测试的上下文中,但 AndroidTestCase 为被测系统提供了一个上下文,它无法从测试包中看到资产。

这是根据给出的答案工作的修改后的代码:

public class AppDescLoaderTest extends InstrumentationTestCase
{
   ...
  protected void setUp() throws Exception
  {
    super.setUp();
    m_app = new Application();
    //call to system under test to load m_app using
    //a sample xml file
     m_appDescLoader = new AppDescLoader(m_app, SAMPLE_XML, getInstrumentation().getContext());
  }
4

3 回答 3

70

选项 1:使用InstrumentationTestCase

假设您在 android 项目和测试项目中都有 assets 文件夹,并将 XML 文件放在 assets 文件夹中。在测试项目下的测试代码中,这将从 android 项目资产文件夹中加载 xml:

getInstrumentation().getTargetContext().getResources().getAssets().open(testFile);

这将从测试项目资产文件夹中加载 xml:

getInstrumentation().getContext().getResources().getAssets().open(testFile);

选项 2:使用类加载器

在您的测试项目中,如果将 assets 文件夹添加到项目构建路径(在 r14 版本之前由 ADT 插件自动完成),您可以在没有上下文的情况下从 res 或 assets 目录(即项目构建路径下的目录)加载文件:

String file = "assets/sample.xml";
InputStream in = this.getClass().getClassLoader().getResourceAsStream(file);
于 2012-03-27T23:15:43.400 回答
9

对于 Android 和 JVM 单元测试,我使用以下内容:

public final class DataStub {
    private static final String BASE_PATH = resolveBasePath(); // e.g. "./mymodule/src/test/resources/";

    private static String resolveBasePath() {
        final String path = "./mymodule/src/test/resources/";
        if (Arrays.asList(new File("./").list()).contains("mymodule")) {
            return path; // version for call unit tests from Android Studio
        }
        return "../" + path; // version for call unit tests from terminal './gradlew test'
    }

    private DataStub() {
        //no instances
    }

    /**
     * Reads file content and returns string.
     * @throws IOException
     */
    public static String readFile(@Nonnull final String path) throws IOException {
        final StringBuilder sb = new StringBuilder();
        String strLine;
        try (final BufferedReader reader = new BufferedReader(new InputStreamReader(new FileInputStream(path), "UTF-8"))) {
            while ((strLine = reader.readLine()) != null) {
                sb.append(strLine);
            }
        } catch (final IOException ignore) {
            //ignore
        }
        return sb.toString();
    }
}

我放入下一个路径的所有原始文件:".../project_root/mymodule/src/test/resources/"

于 2015-10-10T18:33:13.217 回答
1

为 Kotlin 试试这个:

val json = File("src\\main\\assets\\alphabets\\alphabets.json").bufferedReader().use { it.readText() }
于 2020-05-01T07:59:02.687 回答