iOS регулирует асинхронные вызовы API с помощью NSBlockOperation
Я хочу ограничить количество внутренних вызовов API до 2. Я могу создать NSOperationQueue
и добавить блок в очередь, однако каждый вызов API имеет блок завершения, поэтому начальные вызовы ограничены, но я не знаю, как ограничить обработку очереди на основе выполнения блока завершения.
В приведенном ниже коде можно иметь более двух API вызовов в полете в любое время.
NSOperationQueue *requestQueue = [[NSOperationQueue alloc] init];
service.requestQueue.maxConcurrentOperationCount = 2;
for (int i = 0; i < 100; i++)
{
NSBlockOperation *operation = [NSBlockOperation blockOperationWithBlock:^{
[self invokeAPI:kAPIName completion:^BOOL(APIResult *result) {
// Do stuff
}
[requestQueue addOperation:operation];
}
}
Был бы признателен за любые указатели на правильный шаблон для использования.
РЕДАКТИРОВАТЬ - На основе ответа Марк-Александр
Создан этот класс для инкапсуляции операций. Этот подход безопасен с точки зрения памяти, учитывая, что этот класс будет создан из dataAccessService и внедрен, а также блок завершения, имеющий ссылки на self, и конец, вызываемый перед блок завершения выполняется?
@interface MAGApiOperation : NSOperation
@property (nonatomic, strong) id<MAGDataAccessServiceProtocol> dataAccessService;
@property (nonatomic, copy) NSString *apiName;
@property (nonatomic, copy) BOOL (^onCompletion)(APIResult *);
+ (instancetype)apiOperationWithName:(NSString *)apiName dataAccessService:(id<MAGDataAccessServiceProtocol>)dataAccessService completion:(BOOL (^)(APIResult *))onCompletion;
@implementation MAGApiOperation
@synthesize executing = _isExecuting;
@synthesize finished = _isFinished;
#pragma mark - Class methods
/// Creates a new instance of MAGApiOperation
+ (instancetype)apiOperationWithName:(NSString *)apiName dataAccessService:(id<MAGDataAccessServiceProtocol>)dataAccessService completion:(BOOL (^)(APIResult *))onCompletion {
MAGApiOperation *operation = [[self alloc] init];
operation.apiName = apiName;
operation.dataAccessService = dataAccessService;
operation.onCompletion = onCompletion;
return operation;
}
#pragma mark - NSOperation method overrides
- (void)start {
[self willChangeValueForKey:@"isExecuting"];
_isExecuting = YES;
[self didChangeValueForKey:@"isExecuting"];
if (!self.isCancelled)
{
[self invokeApiWithName:self.apiName completion:self.onCompletion];
}
}
- (void)finish {
[self willChangeValueForKey:@"isExecuting"];
[self willChangeValueForKey:@"isFinished"];
_isExecuting = NO;
_isFinished = YES;
[self didChangeValueForKey:@"isExecuting"];
[self didChangeValueForKey:@"isFinished"];
}
#pragma mark - Private methods
/// Invokes the api with the name then executes the completion block
- (void)invokeApiWithName:(NSString *)apiName completion:(BOOL (^)(VAAInvokeAPIResult *))onCompletion {
[self.dataAccessService invokeAPI:kAPIName completion:^BOOL(APIResult *result) { {
[self finish];
return onCompletion(result);
}];
}
1 ответ
Вам нужно создать подкласс NSOperation, чтобы сделать это.
Вот полная документация, которая объясняет все, как подкласс NSOperation: https://developer.apple.com/reference/foundation/operation
Быстрые заметки:
- Ваша "стартовая" операция будет
invokeAPI
вызов. - Затем в вашем блоке завершения invokeAPI вы помечаете свою собственную операцию как завершенную (см. Ниже, обратите внимание, что вызовы willChange и didChange действительно важны)
[self willChangeValueForKey:@"isExecuting"];
[self willChangeValueForKey:@"isFinished"];
self.isExecuting = NO;
self.isFinished = YES;
[self didChangeValueForKey:@"isExecuting"];
[self didChangeValueForKey:@"isFinished"];