测试使用ActionBarSherlock的项目

时间:2012-04-25 10:01:16

标签: android ant actionbarsherlock android-library android-testing

我的设置:

  1. 图书馆项目:ActionBarSherlock
  2. 项目
  3. 测试项目
  4. 我的项目将库项目链接为库项目。它编译并运行良好。

    现在我尝试使用普通测试项目测试我的应用程序。在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);
        }
    }
    

    我可以解决一些问题吗?

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

1 个答案:

答案 0 :(得分:7)

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

首先,请注意Library和“library”之间的区别,其中Library是Android团队规定的名词。我也在使用Referencing项目来描述一个使用Library项目的项目。

即,Referencing项目使用Library项目。

Library“图书馆”项目

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

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

图书馆计划

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

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

除了类依赖关系解析之外,资源也被直接引用并编译到R.java项目Referencing目录中的gen文件中。

新的ADT向人们提出了问题,因为它为包含 Library 项目引用的jar添加了“支持”:

Pre ADT 17

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

ADT 17以后

Dealing With Dependencies

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

不幸的是,只是从ReferencingLibrary项目中移除库(因此只存在一个链接)然后混淆了eclipse,它再也无法看到使用它的项目范围内的jar

要解决此问题,您需要将Library个项目放在/libs目录中 - 如果您的硬盘位于硬盘驱动器的不同位置,这可能会让您感到痛苦。然后,这些广告将自动用于您的LibraryReferencing项目。

所以,要超越ADT 17:

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