Android Mockito 单元测试中的意图附加功能

问题描述 投票:0回答:2

我试图验证特定的额外内容是否已添加到 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"));

    }
}

您对我应该如何进行此测试有什么建议吗? 预先感谢

android unit-testing android-intent mockito extras
2个回答
12
投票
默认情况下,

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'

0
投票

您可以使用 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
}
© www.soinside.com 2019 - 2024. All rights reserved.