我被通缉但没有调用。在第 verify(fingerPrintHelper, times(1)).initializeFingerPrint(any());
行checkFingerPrintWhenTouchIdEnabled()
方法的模拟错误与此模拟错误的交互为零 . 甚至我也嘲笑了对象,并且在调试时调用initializeFingerPrint(..)
函数。
下面是我要测试的功能,
@RequiresApi(Build.VERSION_CODES.M)
public void checkFingerPrint() {
if (fingerPrintHelper.isDeviceReadyForFingerPrint()) {
boolean isFingerPrintEnable = sharedPreference.getBoolean(SpKeys.KEY_FINGER_PRINT, false);
if (isFingerPrintEnable) {
fingerPrintHelper.initializeFingerPrint(this);
}
} else {
sharedPreference.putBoolean(SpKeys.KEY_FINGER_PRINT, false).commit();
}
}
登录行动.java
public class LoginActivity extends AppCompatActivity {
public FingerPrintHelper fingerPrintHelper;
ActivityLoginBinding binding;
private LoginViewModel loginViewModel;
private SharedPreferenceManager sharedPreferenceManager;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
this.requestWindowFeature(Window.FEATURE_NO_TITLE);
sharedPreferenceManager = new SharedPreferenceManager(getApplicationContext(), SpKeys.MY_SP);
fingerPrintHelper = new FingerPrintHelper(this);
binding = DataBindingUtil.setContentView(this, R.layout.activity_login);
loginViewModel = ViewModelProviders.of(this).get(LoginViewModel.class);
binding.setViewModel(loginViewModel);
binding.setLifecycleOwner(this);
checkFingerPrint();
}
@RequiresApi(Build.VERSION_CODES.M)
public void checkFingerPrint() {
if (fingerPrintHelper.isDeviceReadyForFingerPrint()) {
boolean isFingerPrintEnable = sharedPreferenceManager.getBoolean(SpKeys.KEY_FINGER_PRINT, false);
if (isFingerPrintEnable) {
fingerPrintHelper.initializeFingerPrint(this);
}
} else {
sharedPreferenceManager.setBoolean(SpKeys.KEY_FINGER_PRINT, false);
}
}
}
我正在为此函数编写正面和负面测试用例checkFingerPrintWhenTouchIdDisabled()
而测试工作正常,但checkFingerPrintWhenTouchIdEnabled()
测试功能出现错误请参考以下测试类
登录活动测试.java
public class LoginActivityTest {
LoginActivity loginActivity;
@Mock
FingerPrintHelper fingerPrintHelper;
@Rule
public ActivityTestRule<LoginActivity> loginActivityRule = new ActivityTestRule<>(
LoginActivity.class);
Context context;
@Before
public void setUp() {
MockitoAnnotations.initMocks(this);
loginActivity = loginActivityRule.getActivity();
context = getInstrumentation().getTargetContext();
}
@Test
public void checkFingerPrintWhenTouchIdDisabled() {
SharedPreferences sharedPreferences = context.getSharedPreferences(SpKeys.MY_SP, Context.MODE_PRIVATE);
when(fingerPrintHelper.isDeviceReadyForFingerPrint()).thenReturn(false);
loginActivity.checkFingerPrint();
Assert.assertFalse(sharedPreferences.getBoolean(SpKeys.KEY_FINGER_PRINT, false));
verify(fingerPrintHelper, never()).initializeFingerPrint(any());
}
@Test
public void checkFingerPrintWhenTouchIdEnabled() {
SharedPreferences sharedPreferences = context.getSharedPreferences(SpKeys.MY_SP, Context.MODE_PRIVATE);
SharedPreferences.Editor preferencesEditor = sharedPreferences.edit();
when(fingerPrintHelper.isDeviceReadyForFingerPrint()).thenReturn(true);
preferencesEditor.putBoolean(SpKeys.KEY_FINGER_PRINT, true).commit();
loginActivity.checkFingerPrint();
/* Below verification for `initializeFingerPrint()` is throwing error as,
Actually, there were zero interactions with this mock. error,
Even I have mock the object & while debugging the method is getting invoked also.
If I debug my code it calls this function but in test cases it shows above error
*/
verify(fingerPrintHelper, times(1)).initializeFingerPrint(any());
}
}
那为什么我的测试用例会出现零交互错误呢?可能是什么问题,任何帮助都值得赞赏。
提前谢谢。
尝试以下操作来手动设置模拟(不使用注释(:
@Before
public void setUp() {
loginActivity = loginActivityRule.getActivity();
loginActivity.fingerPrintHelper = Mockito.mock(FingerPrintHelper.class);
// ...
}
如果以前可以成功创建loginAcitivy
,那么您现在应该不会遇到问题。
无论如何,fingerPrintHelper
似乎都是public
的,所以很容易设置。
但是如果你想正确地做到这一点,你可以提供一个二传手。
或者,如果您想保留注释以创建fingerPrintHelper
.
@Mock
FingerPrintHelper fingerPrintHelper;
@Before
public void setUp() {
MockitoAnnotations.initMocks(this);
loginActivity = loginActivityRule.getActivity();
loginActivity.fingerPrintHelper = fingerPrintHelper;
// ...
}
我仍然想知道保持loginActivity.fingerPrintHelper = fingerPrintHelper行的原因。
模拟不会神奇地将自己附加到任何其他对象上。
@InjectMocks
会为你做到这一点,但Mockito似乎无法独自处理你的LoginActivity
的创建。
因此,您唯一能做的就是手动将模拟传递给被测试的对象。
您没有在测试用例中的任何位置注入模拟。我假设在构造函数/工厂中创建了一个普通实例。
要么使用 SUT 的二传手,要么让 Mockito 为您注入它:
@InjectMocks
LoginActivity loginActivity;
仅使用@Mock
是不够的。