我在浓缩咖啡测试中遇到问题,我不知道为什么匹配文本总是失败,我什至尝试创建具有两个活动的简单应用程序,第一个活动具有textview和两个按钮,一个按钮显示吐司进行下一个活动,在录制过程中,向文本中添加断言并正确生成代码时,我运行测试,但始终无法显示此错误:
android.support.test.espresso.NoMatchingViewException:找不到层次结构中符合以下条件的视图:(标识为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));
}
};
}
}
这是第一个活动的布局:
<?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>
第二个活动只有一个textview,如果我在其上添加断言,它也会失败。
如果我评论测试将通过的文本断言,但是如果文本断言在那里通过,则它总是会失败,我还在每个步骤之后添加睡眠(我知道它与文本无关,但以防万一),我无法理解该代码是由记录生成的,并且文本可以正确显示,但显示找不到层次结构中的视图匹配!
我做错了什么? 预先感谢
答案 0 :(得分:1)
以下代码看起来完全错误:
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")));
根据您的布局,只有一个TextView
组件没有任何子组件。因此,childAtPosition()
无法应用。
textView.check(matches(withText("Rooh")));
是多余的,因为在声明textView
时已经这样做了。
您可以将代码修改为以下有效的代码:
ViewInteraction textView = onView(allOf(withId(R.id.textView), withText("Rooh")));
textView.check(matches(isDisplayed()));
有关匹配文本的更多信息,您可以找到here。
答案 1 :(得分:1)
此问题归因于断言依赖于UI Automator层次结构来建立元素的子位置,而动作依赖于Espresso层次结构。在重播期间,动作和断言都是使用Espresso层次结构执行的,因此,由于UI Automator和Espresso层次结构之间的子位置不匹配,断言更容易失败。
有几种解决此限制的方法:
1)在Android Studio中,在文件|设置,打开构建,执行,部署| Espresso Test Recorder并将“ Assertion depth”设置为1。这样,添加的断言将不依赖于子位置来标识已断言的元素(为断言生成的测试代码看起来与上面的答案中所建议的非常相似) 。缺点是断言可能变得太笼统,可能会匹配多个元素。
2)依靠动作来断言屏幕上某些元素的存在,例如,不要断言屏幕上存在带有特定文本的按钮,而是单击它。
3)仅记录动作并将声明手动添加到生成的Espresso测试中。
4)手动调整记录的断言以使用正确的子位置。