我的设置:
- 图书馆项目:动作栏夏洛克
- 项目
- 测试项目
链接为库项目。它可以编译并运行良好。
现在,我尝试使用普通测试项目测试我的应用程序。在 eclipse 中运行测试效果很好。如果我尝试使用 ant 运行测试,测试项目甚至不会编译:
[javac] LoginActivityTest.java:9: cannot access com.actionbarsherlock.app.SherlockActivity
[javac] class file for com.actionbarsherlock.app.SherlockActivity not found
[javac] public class LoginActivityTest extends ActivityInstrumentationTestCase2<LoginActivity> {
[javac] ^
[javac] LoginActivityTest.java:25: cannot find symbol
通过 eclipse 构建工作完美,测试运行也完美。
如果我将库项目链接到我的测试项目,它会使用 ant 编译,但测试失败。
[exec] Error in testSuiteConstructionFailed:
[exec] java.lang.RuntimeException: Exception during suite construction
[exec] at android.test.suitebuilder.TestSuiteBuilder$FailedToCreateTests.testSuiteConstructionFailed(TestSuiteBuilder.java:238)
[exec] at java.lang.reflect.Method.invokeNative(Native Method)
[exec] at android.test.AndroidTestRunner.runTest(AndroidTestRunner.java:169)
[exec] at android.test.AndroidTestRunner.runTest(AndroidTestRunner.java:154)
[exec] at android.test.InstrumentationTestRunner.onStart(InstrumentationTestRunner.java:537)
[exec] at android.app.Instrumentation$InstrumentationThread.run(Instrumentation.java:1551)
[exec] Caused by: java.lang.reflect.InvocationTargetException
[exec] at java.lang.reflect.Constructor.constructNative(Native Method)
[exec] at java.lang.reflect.Constructor.newInstance(Constructor.java:417)
[exec] at android.test.suitebuilder.TestMethod.instantiateTest(TestMethod.java:87)
[exec] at android.test.suitebuilder.TestMethod.createTest(TestMethod.java:73)
[exec] at android.test.suitebuilder.TestSuiteBuilder.addTest(TestSuiteBuilder.java:262)
[exec] at android.test.suitebuilder.TestSuiteBuilder.build(TestSuiteBuilder.java:184)
[exec] at android.test.InstrumentationTestRunner.onCreate(InstrumentationTestRunner.java:371)
[exec] at com.zutubi.android.junitreport.JUnitReportTestRunner.onCreate(JUnitReportTestRunner.java:90)
[exec] at android.app.ActivityThread.handleBindApplication(ActivityThread.java:3891)
[exec] at android.app.ActivityThread.access$1300(ActivityThread.java:122)
[exec] at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1184)
[exec] at android.os.Handler.dispatchMessage(Handler.java:99)
[exec] at android.os.Looper.loop(Looper.java:137)
[exec] at android.app.ActivityThread.main(ActivityThread.java:4340)
[exec] at java.lang.reflect.Method.invokeNative(Native Method)
[exec] at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:784)
[exec] at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:551)
[exec] at dalvik.system.NativeStart.main(Native Method)
[exec] Caused by: java.lang.NoClassDefFoundError: com.myproject.android.app.activities.LoginActivity
[exec] at com.myproject.android.app.test.LoginActivityTest.<init>(LoginActivityTest.java:18)
[exec] ... 19 more
我的测试类:
public class LoginActivityTest extends ActivityInstrumentationTestCase2<LoginActivity> {
private LoginActivity mActivity;
private EditText mTextUserName;
private EditText mTextUserPassword;
public LoginActivityTest() {
// the super call is line 18 (see stack trace above)
super("com.myproject.android.app.activities", LoginActivity.class);
}
@Override
protected void setUp() throws Exception {
super.setUp();
mActivity = getActivity();
mTextUserName = (EditText) mActivity.findViewById(com.myproject.android.app.R.id.login_activity_username);
mTextUserPassword = (EditText) mActivity.findViewById(com.myproject.android.app.R.id.login_activity_password);
}
public void testPreConditions() {
assertTrue("Activity is null!", mActivity != null);
}
public void testLogin() throws Throwable {
mActivity.runOnUiThread(new Runnable() {
public void run() {
mTextUserName.setText("username");
mTextUserPassword.setText("password");
}
});
sendKeys(KeyEvent.KEYCODE_ENTER);
}
}
一些想法如何解决这个问题?
更新:看起来蚂蚁构建/测试仍然一团糟。根据这篇关于测试库项目的博客文章,列出的 7 个问题中的大多数将在下一个 ADT 版本 (ADT r20) 中修复。
自从 ADT 17 破坏/修复所有内容以来,关于使用Library
项目有很多不同的信息(取决于您目前是否用头撞桌子)。
首先,请注意Library
和"库"之间的区别,其中Library
是Android团队规定的名词。我也在使用Referencing
描述使用Library
项目的项目的项目。
即,Referencing
项目使用Library
项目。
非Library
"图书馆"项目
在正常的 Java 开发中,可以在 Eclipse 中链接一个依赖于另一个源的项目。我相信这种方法具有使用图书馆项目源作为参考项目源的效果。这意味着在编译引用项目时,库项目源在范围内。库项目类与引用项目的类同时生成。
当构建用于大多数情况下时,这非常有效,因为所有类都是构建的,然后打包在 JAR 或 WAR 或其他什么中。
图书馆项目
库项目的竞争(不是混合和匹配)方法是 Android 团队Library
项目:
标记为Library
项目的 Android 项目会将其源代码编译并构建到其 bin
目录中的 jar
文件中(在 clean/build 命令之后)。任何Referencing
项目都会自动导入此jar
并访问Library
项目功能。您可以通过查看包资源管理器中的 Java 库Android Dependencies
来查看此关系。
除了类依赖关系解析之外,资源还被直接引用并编译到Referencing
项目gen
目录中的R.java
文件中。
新ADT
给人们设置带来了问题,因为它增加了"支持",包括Library
项目引用的罐子:
ADT 17 之前
Library
项目在其构建路径中添加了一个罐子。Referencing
项目还可以将 jar 添加到其自己的构建路径中。
ADT 17 及以上
处理依赖关系
在ADT 17之后,Library
动态引用自己的罐子的项目开始表现得很奇怪。这不仅仅是在Referencing
项目中包含相同的依赖引用以使jar在两个作用域中都可见的情况。现在,这导致了所包含类的奇怪重复。
不幸的是,简单地从Referencing
或Library
项目中删除库(因此只有一个链接存在)然后混淆了 eclipse,它无法再从使用它的项目范围内看到 jar。
要解决此问题,您需要将Library
项目 jar 放在 /libs
目录中 - 如果您的 jar 位于硬盘驱动器上的不同位置,这可能会很痛苦。然后,这些 jar 将自动用于您的Library
和Referencing
项目。
因此,要超越 ADT 17:
- 删除当前项目范围内未用作源的任何 jar 文件(无论是
Library
还是Referencing
项目)。 - 从
Library
项目中删除任何"库"项目(而是将它们编译到 jar 中)。 - 从
Library
项目中删除外部 jar,将它们复制到libs
目录中。