java - 使用 AlarmManager 测试 BroadcastReceiver

标签 java android junit broadcastreceiver alarmmanager

我正在尝试使用 AlarmManager 测试启动警报并使用 BroadcastReceiver 接受警报。我找到了这个sources ,并像那里那样做一切。但它工作得一般,并且时不时地触发(例如 20 次尝试中有 1 次成功):

@Test public void testOnReceive_defaultAlarm_accepted() throws Exception {

    final SharedPreferences sharedPrefs = mQuestions.getAppComponent().getSharedPrefs();

    final Calendar calendar = Calendar.getInstance();

    calendar.setTimeInMillis(System.currentTimeMillis() + TIME_DELAY_SNOOZE); // time for alarm

    sharedPrefs.edit()
        .putString(mQuestions.getString(R.string.pref_notification_time_key),
            TimePreference.timeToString(calendar.get(Calendar.HOUR), calendar.get(Calendar.MINUTE)))
        .apply(); // set test prefs to alarm

    QuestionNotificationUtils.launchDefaultAlarm(mQuestions, sharedPrefs); // launch alarm

    Thread.sleep(TIME_DELAY_SNOOZE + TIME_DELAY); // wait for alarm to be fired

    assertNotNull(mQuestionNotificationReceiver.mRealmWrapper); // check if injected, and HERE IT FAILS

    verify(mQuestionNotificationReceiver.mRealmWrapper, times(1)).getQuestions(); // check if fired with Mockito
}

启动默认警报是围绕此函数的包装:

private static void launchAlarm(Context ctx, SharedPreferences prefs, int id) {
    final AlarmManager alarmMgr = (AlarmManager) ctx.getSystemService(Context.ALARM_SERVICE);
    final PendingIntent alarmIntent =
        PendingIntent.getBroadcast(ctx, id, QuestionNotificationReceiver.getIntent(ctx, id),
            PendingIntent.FLAG_UPDATE_CURRENT);

    final Calendar calendar = Calendar.getInstance();
    calendar.setTimeInMillis(System.currentTimeMillis());

    final String time = prefs.getString(ctx.getString(R.string.pref_notification_time_key),
        ctx.getString(R.string.questions_preference_fragment_default_notification_time));

    calendar.set(Calendar.HOUR_OF_DAY, TimePreference.parseHour(time));
    calendar.set(Calendar.MINUTE, TimePreference.parseMinute(time));

    alarmMgr.setRepeating(AlarmManager.RTC, calendar.getTimeInMillis(), AlarmManager.INTERVAL_DAY,
        alarmIntent);
}

广播接收器的部分:

@Inject RealmWrapper mRealmWrapper;

@Override public void onReceive(Context context, Intent intent) {

    ((Questions) context.getApplicationContext()).getQuestionNotificationReceiverComponent()
        .inject(this);

    onPostInjectReceive(context, intent);
}

这里还有@Before方法:

@Before public void setUp() throws Exception {

    mQuestionNotificationReceiver = new QuestionNotificationReceiver();

    mQuestions.registerReceiver(mQuestionNotificationReceiver,
        new IntentFilter(QuestionNotificationReceiver.getAction(mQuestions)));
}

有时它会起作用,它表明,代码几乎是正确的,但它非常非常不稳定。如果我开始使用自定义 RetryRule 来重试失败的测试,测试会变得非常长,因为 Thread.sleep() 需要大约 10 秒。如何应对这种情况?

最佳答案

使用 Calendar.HOUR 或 Calendar.HOUR_OF_DAY,而不是同时使用

关于java - 使用 AlarmManager 测试 BroadcastReceiver,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39680664/

相关文章:

java - 使用 WrapLayout/FlowLayout 时是否可以水平对齐组件?

java - openGL/JOGL : Why do some Textures render "Smoothed" and some not?

android - 选项菜单样式

java - Android 的换行小部件布局

使用 Microsoft Visual Studio Code 进行 Java 编程

Android 6.0 Marshmallow BLE 连接问题

java - 在 test() 函数中执行测试之前,调用 setUp() 中另一个类的所有静态方法,包括 main

java - 将单元测试标记为 JUnit 中的预期失败

java - TDD:我是否必须定义我的代码不应该做的所有事情?

Java泛型 <?扩展 A> vs <A> vs <?超A>