我试图验证特定的额外内容是否已添加到 Intent 中,但我在 Android 单元测试中的 Intent 始终为 null。我有以下课程需要测试:
public class TestClass extends SomeNotifier {
private Intent mIntent = new Intent("testintent");
public TestClassConstructor(Context context) {
super(context);
}
@Override
public void notifyChange() {
mIntent.putExtra("Key one", 22);
mIntent.putExtra("Key two", 23);
mIntent.putExtra("Key three", 24);
getContext().sendBroadcast(mIntent);
}
}
测试如下(我也尝试使用mockIntent,但结果是相同的,额外的内容再次为空):
@RunWith(MockitoJUnitRunner.class)
public class TestClassTest {
@Mock
Context mMockContext;
@Test
public void sendBroadcastTest() {
ArgumentCaptor<Intent> argument = ArgumentCaptor.forClass(Intent.class);
TestClass testClassNotifier = new TestClass (mMockContext);
testClassNotifier.notifyChange();
verify(mMockContext).sendBroadcast(argument.capture());
Intent intent = argument.getValue();
//THE INTENT IS NULL
Assert.assertTrue(intent.hasExtra("Key one"));
}
}
您对我应该如何进行此测试有什么建议吗? 预先感谢
Intent
和其他 Android 运行时类(如 Context
)仅在物理 Android 手机和模拟器上可用。对于本地单元测试,您将获得运行时的存根版本,其中 Intent
等方法默认返回 null
。请参阅此问题以获取解释。
这意味着您当前的测试正在验证
Intent
的存根版本,默认情况下将为所有方法调用返回 null
。
有一些方法可以解决这个问题 - 您可以将测试转换为仪器化单元测试,这意味着您必须在真实的手机或模拟器上运行它。或者,您可以用您控制的类包装
Intent
类。
也许最好的解决方案是使用像Robolectric这样的框架。这将为您提供重要 Android 类的工作测试替身(称为影子),例如
Intent
,用于在 IDE 中运行的本地单元测试。
仔细安装 Robolectric 后,您只需添加以下内容即可进行测试:
@RunWith(RobolectricTestrunner.class)
public class TestClassTest {
顺便说一句,请确保您在 Android 上使用 Mockito 的正确依赖项:
testCompile 'org.mockito:mockito-core:2.8.9'
androidTestCompile 'org.mockito:mockito-android:2.8.9'
您可以使用 Mockk 进行模拟
intent
@Test
fun `Test`() {
// Arrange
val longValue = 1000L
val enumValue = ModelEnum.MODEL1
val intentMock = spyk<Intent>()
every { intentMock.extras?.getLong("KEY_LONG_VALUE") } returns longValue
every { intentMock.hasExtra(CheckoutRouter.RESULT_PAYMENT_TYPE) } returns true
every { intentMock.getSerializableExtra("KEY_ENUM_VALUE") } returns enumValue
// Action
val a = A(intent = intentMock)
val longValueResult = a.getLongValue()
val enumValueResult = a.getEnumValue()
// Assert
Assert.assertEquals(longValue, longValueResult)
Assert.assertEquals(enumValue, enumValueResult)
}
class A (val intent: Intent){
fun getLongValue():Long? {
return intent.extras?.getLong("KEY_LONG_VALUE")
}
fun getEnumValue():ModelEnum? {
return intent.getSerializableExtra("KEY_ENUM_VALUE") as? ModelEnum
}
}
enum class ModelEnum{
MODEL1,MODEL2
}