Выполните действие в час, каждый час, с помощью ReactiveCocoa
Попытка следовать рекомендациям ReactiveCocoa для обновления моего интерфейса в час, каждый час. Это то, что у меня есть:
NSDateComponents *components = [[[NSCalendar sharedCalendar] calendar] components:NSMinuteCalendarUnit fromDate:[NSDate date]];
// Generalization, I know (not every hour has 60 minutes, but bear with me).
NSInteger minutesToNextHour = 60 - components.minute;
RACSubject *updateEventSignal = [RACSubject subject];
[updateEventSignal subscribeNext:^(NSDate *now) {
// Update some UI
}];
[[[RACSignal interval:(60 * minutesToNextHour)] take:1] subscribeNext:^(id x) {
[updateEventSignal sendNext:x];
[[RACSignal interval:3600] subscribeNext:^(id x) {
[updateEventSignal sendNext:x];
}];
}];
У этого есть некоторые очевидные недостатки: ручная подписка и отправка, и это просто "чувствует себя не так". Любые идеи о том, как сделать это более "реактивным"?
Ответы
Ответ 1
Вы можете сделать это, используя полностью ванильные операторы. Это просто вопрос объединения двух интервалов вместе, все еще проходящих через оба их значения, что именно то, что - concat: делает.
Я бы переписал тему следующим образом:
RACSignal *updateEventSignal = [[[RACSignal
interval:(60 * minutesToNextHour)]
take:1]
concat:[RACSignal interval:3600]];
Это может не дать вам сверхточную точность (потому что между двумя сигналами может быть небольшая икота), но это должно быть Good Enough ™ для любого пользовательского интерфейса.
Ответ 2
Похоже, вам нужно что-то вроде +interval:startingIn:
.
С этой мыслью вы можете сделать свою собственную версию +interval:startingIn:
, слегка изменив реализацию +interval:
.
+ (RACSignal *)interval:(NSTimeInterval)interval startingIn:(NSTimeInterval)delay {
return [[RACSignal createSignal:^(id<RACSubscriber> subscriber) {
int64_t intervalInNanoSecs = (int64_t)(interval * NSEC_PER_SEC);
int64_t delayInNanoSecs = (int64_t)(delay * NSEC_PER_SEC);
dispatch_source_t timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0));
dispatch_source_set_timer(timer, dispatch_time(DISPATCH_TIME_NOW, delayInNanoSecs), (uint64_t)intervalInNanoSecs, (uint64_t)0);
dispatch_source_set_event_handler(timer, ^{
[subscriber sendNext:[NSDate date]];
});
dispatch_resume(timer);
return [RACDisposable disposableWithBlock:^{
dispatch_source_cancel(timer);
dispatch_release(timer);
}];
}] setNameWithFormat:@"+interval: %f startingIn: %f", (double)interval, (double)delay];
}
С помощью этого кода ваш код может быть реорганизован на:
NSDateComponents *components = [[[NSCalendar sharedCalendar] calendar] components:NSMinuteCalendarUnit fromDate:[NSDate date]];
// Generalization, I know (not every hour has 60 minutes, but bear with me).
NSInteger minutesToNextHour = 60 - components.minute;
RACSubject *updateEventSignal = [[RACSignal interval:3600 startingIn:(minutesToNextHour * 60)];
[updateEventSignal subscribeNext:^(NSDate *now) {
// Update some UI
}];