android espresso 测试总是在文本匹配中失败

android espresso test is fails always in text matching

我在 espresso 测试中遇到问题,我不知道为什么匹配文本总是失败,我什至尝试创建简单的应用程序有两个活动,第一个 activity 有 textview 和两个按钮一个按钮显示敬酒另一个下一步 activity,在记录期间,当向文本添加断言并且正确生成代码时,我 运行 测试它总是失败并显示此错误:

android.support.test.espresso.NoMatchingViewException:在层次结构中未找到匹配的视图:(id:com.example.admin.testtest2demoo:id/textView 和文本:是 "Rooh" 并且子项位于位置 0在 id 为 android:id/content 的父子位置 0 中,并显示在屏幕上给用户)

这里是测试代码:

package com.example.admin.testtest2demoo; 

@LargeTest
@RunWith(AndroidJUnit4.class)
public class MainActivityTest {

@Rule
public ActivityTestRule<MainActivity> mActivityTestRule = new ActivityTestRule<>(MainActivity.class);

@Test
public void mainActivityTest() {

    try {
        Thread.sleep(2000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    }

    ///when click on this button show toast
    ViewInteraction appCompatButton = onView(
            allOf(withId(R.id.button2), withText("show SomeThing"),
                    childAtPosition(
                            childAtPosition(
                                    withId(android.R.id.content),
                                    0),
                            2),
                    isDisplayed()));
    appCompatButton.perform(click());

    try {
        Thread.sleep(2000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    }

    ///when click on this button show toast
    ViewInteraction appCompatButton2 = onView(
            allOf(withId(R.id.button2), withText("show SomeThing"),
                    childAtPosition(
                            childAtPosition(
                                    withId(android.R.id.content),
                                    0),
                            2),
                    isDisplayed()));
    appCompatButton2.perform(click());

    try {
        Thread.sleep(2000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    }

    ///check the text in first activity (is it Rooh)
    ViewInteraction textView = onView(
            allOf(withId(R.id.textView), withText("Rooh"),
                    childAtPosition(
                            childAtPosition(
                                    withId(android.R.id.content),
                                    0),
                            0),
                    isDisplayed()));
    textView.check(matches(withText("Rooh")));

    try {
        Thread.sleep(2000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    }

    ///when click on this button go to next activity
    ViewInteraction appCompatButton3 = onView(
            allOf(withId(R.id.button), withText("Go Next"),
                    childAtPosition(
                            childAtPosition(
                                    withId(android.R.id.content),
                                    0),
                            0),
                    isDisplayed()));
    appCompatButton3.perform(click());

    try {
        Thread.sleep(2000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    }

    ///check the text in second activity (is it Hello World!)
    ViewInteraction textView2 = onView(
            allOf(withId(R.id.tv1), withText("Hello World!"),
                    childAtPosition(
                            allOf(withId(R.id.background),
                                    childAtPosition(
                                            withId(R.id.swipe_refresh_layout),
                                            0)),
                            0),
                    isDisplayed()));
    textView2.check(matches(withText("Hello World!")));
}

private static Matcher<View> childAtPosition(
        final Matcher<View> parentMatcher, final int position) {

    return new TypeSafeMatcher<View>() {
        @Override
        public void describeTo(Description description) {
            description.appendText("Child at position " + position + " in parent ");
            parentMatcher.describeTo(description);
        }

        @Override
        public boolean matchesSafely(View view) {
            ViewParent parent = view.getParent();
            return parent instanceof ViewGroup && parentMatcher.matches(parent)
                    && view.equals(((ViewGroup) parent).getChildAt(position));
        }
    };
}
}

这是第一个 activity 的布局:

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout 
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity">


<Button
    android:id="@+id/button"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:layout_below="@+id/button2"
    android:onClick="goNext"
    android:text="Go Next"
    app:layout_constraintBottom_toBottomOf="parent"
    app:layout_constraintEnd_toEndOf="parent"
    app:layout_constraintStart_toStartOf="parent"
    app:layout_constraintTop_toBottomOf="@+id/button2"
    app:layout_constraintVertical_bias="0.183" />

<TextView
    android:id="@+id/textView"
    android:layout_width="97dp"
    android:layout_height="27dp"
    android:layout_marginTop="80dp"
    android:gravity="center"
    android:text="Rooh"
    app:layout_constraintEnd_toEndOf="parent"
    app:layout_constraintHorizontal_bias="0.501"
    app:layout_constraintStart_toStartOf="parent"
    app:layout_constraintTop_toTopOf="parent" />

<Button
    android:id="@+id/button2"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:layout_below="@+id/textView"
    android:layout_marginTop="64dp"
    android:onClick="showSomeThing"
    android:text="show SomeThing"
    app:layout_constraintEnd_toEndOf="parent"
    app:layout_constraintHorizontal_bias="0.502"
    app:layout_constraintStart_toStartOf="parent"
    app:layout_constraintTop_toBottomOf="@+id/textView" />
</android.support.constraint.ConstraintLayout>

而第二个 activity 只有一个文本视图,如果我在其上添加断言,它也会失败。

如果我评论测试将通过的文本断言,但如果有文本断言它总是失败,我还在每一步之后包括睡眠(我知道这与它无关但以防万一),我可以不明白代码是由录音生成的,文本显示正确,但它说在层次结构中找不到匹配的视图!!

我做错了什么?! 提前致谢

下面的代码看起来完全错误:

ViewInteraction textView = onView(
            allOf(withId(R.id.textView), withText("Rooh"),
                    childAtPosition(
                            childAtPosition(
                                    withId(android.R.id.content),
                                    0),
                            0),
                    isDisplayed()));
    textView.check(matches(withText("Rooh")));
  1. 根据您的布局,只有一个 TextView 组件没有任何子组件。所以,childAtPosition()不能应用于它。

  2. textView.check(matches(withText("Rooh"))); 是多余的,因为您在声明 textView.

  3. 时已经这样做了

您可以将代码修改为以下应该有效的代码:

ViewInteraction textView = onView(allOf(withId(R.id.textView), withText("Rooh")));
textView.check(matches(isDisplayed()));

有关匹配文本的更多信息,您可以找到 here

此问题是由于断言依赖 UI Automator 层次结构来建立元素的子位置,而操作依赖于 Espresso 层次结构。在回放期间,操作和断言均使用 Espresso 层次结构执行,因此,由于 UI Automator 和 Espresso 层次结构之间的子位置不匹配,断言更容易失败。

此限制有几种解决方法:

1) 在 Android Studio 中,在文件中 | Settings,打开Build,Execution,Deployment | Espresso Test Recorder 并将 "Assertion depth" 设置为 1。这样,添加的断言将不会依赖子位置来识别断言元素(为断言生成的测试代码看起来与上面的答案中建议的非常相似).缺点是断言可能变得过于笼统,可能匹配多个元素。

2) 依靠动作断言屏幕上某些元素的存在,例如,不是断言屏幕上存在带有特定文本的按钮,而是点击它。

3) 仅记录操作并手动将断言添加到生成的 Espresso 测试中。

4) 手动调整记录的断言以使用正确的子位置。