测试使用ActionBarSherlock的项目

War*_*ith 5 ant android android-library actionbarsherlock android-testing

我的设置:

  1. 图书馆项目:ActionBarSherlock
  2. 项目
  3. 测试项目

我的项目将库项目链接为库项目.它编译并运行良好.

现在我尝试使用正常的测试项目来测试我的应用程序.在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
Run Code Online (Sandbox Code Playgroud)

通过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
Run Code Online (Sandbox Code Playgroud)

我的考试班:

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);
    }
}
Run Code Online (Sandbox Code Playgroud)

一些想法如何解决这个问题?

更新:看起来蚂蚁构建/测试仍然是一团糟.根据这篇关于测试库项目的博客文章,7个列出的问题中的大多数将在下一个ADT版本(ADT r20)中修复.

Gra*_*eme 7

关于使用Library项目有很多不同的信息,因为ADT 17打破/修复了所有内容(取决于你目前是否正在敲打桌面).

首先,请注意Library与"library" 之间的区别,其中Library是由Android团队颁布的名词.我也在使用Referencing描述使用项目的项目的Library项目.

即,一个Referencing项目使用一个Library项目.

Library"图书馆"项目

在正常的Java开发中,可以链接Eclipse中的项目,其中一个项目依赖于另一个项目的源代码.我相信这种方法具有使用库项目源作为引用项目的源的效果.这意味着在编译引用项目时,库项目源在范围内.库项目类与引用项目同时构建.

当在大多数情况下构建以供使用时,这非常有效,因为所有类都是构建的,然后打包在JAR或WAR或其他任何类中.

图书馆项目

Android团队项目是针对图书馆项目的竞争(混合和匹配)方法Library:

标记为项目的Android项目Library将编译并将其源代码构建到jarbin目录中的文件中(在clean/build命令之后).任何Referencing项目都会自动导入jar并获得对Library项目功能的访问权限.您可以通过Android Dependencies在Package Explorer中查看Java Library来查看此关系.

除了类依赖分辨率的资源也被直接引用,并汇编成R.java的内部文件的Referencing项目gen目录.

新的ADT问题引起了人们的关注,因为它增加了"支持",包括Library项目引用的罐子:

ADT前17

Library项目在其构建路径中添加了一个jar.该Referencing项目还可以将jar添加到其自己的构建路径中.

ADT 17以后

处理依赖关系

在ADT 17之后Library,动态引用自己的jar的项目开始表现得很奇怪.这不仅仅是在您的Referencing项目中包含相同的依赖引用以保持jar在两个范围中都可见的情况.这现在导致包含类的奇怪重复.

不幸的是简单地删除从库ReferencingLibrary项目(所以只有一个链接存在),那么困惑日食,它可能不再能看到从项目的使用它的范围罐子.

要解决此问题,您需要将Library项目罐子放在/libs目录中 - 如果您的罐子位于硬盘驱动器的不同位置,这可能是后方的痛苦.这些罐子将自动在您的使用LibraryReferencing项目.

所以,要超越ADT 17:

  • 删除任何在当前项目范围中用作源的jar文件(无论是您的项目Library还是Referencing项目).
  • 从项目中删除所有"库"项目Library(而不是将它们编译成jar).
  • Library项目中删除外部jar ,将它们复制到libs目录中.