可可回调设计:最佳实践

时间:2010-07-03 05:29:34

标签: objective-c cocoa callback

我正在为可可应用程序编写中间件,并且正在讨论如何为许多长时间运行的进程设计回调。

当UI调用一个长时间执行的函数时,它需要提供一个委托,至少允许:

  • 成功报告(返回值)
  • 失败报告(有错误值)
  • 进度报告(已完成,预计总数)

我过去曾尝试过一些技巧,如下所示

@interface MyClass {
}

//Callback Option1, delgate conforming to protocol
-(void) longRunningProcess2:(id<CallbackProtocol>) delegate;

//Callback Option2, provide a delegate and allow it to provide selectors to callback
-(void) longRunningProcess3:(id) delegate success:(SEL) s1 failure:(SEL) s2 progress:(SEL) s3
@end

对于选项1,问题是如何表示委托响应。我考虑的第一种方法是(为简单起见,函数名称是最小的)

//have a generic callback protocol for every function
@protocol CallbackProtocolGeneric
-(void) success:(id) returnValue;
-(void) failure:(NSError*) error;
@optional
-(void) progress:(NSInteger) completed of:(NSInteger) total;
@end

//have a separate protocol for every function
@protocol CallbackProtocolWithObjectAForOperation1
-(void) objectA:(ObjectA*) objectA operation1SucceedWithValue:(ReturnObject*) value;
-(void) objectA:(ObjectA*) objectA operation1FailedWithError:(NSError*) error;
@optional
-(void) objectA:(ObjectA*) objectA operation1didProgress:(NSInteger) completed of:(NSInteger) total;
@end

根据我的经验, 使用通用协议回调选项1 很难使用,因为如果某个类想要成为多个操作的回调,则无法区分它正在接收哪个回调。

回调选项2 使用起来很麻烦,并且使用起来不自然。另外,如果协议被扩展,则需要修改每个呼叫。

每个进程使用特定协议

回调选项1 似乎是最具可读性和可扩展性的方法,但我不知道是否为每个功能制定新协议详细(假设一个给定的对象有10个以上的“长操作”,然后是10个不同的协议)。

在实施此类设计时,其他人会得出什么结论?

- 编辑: 回复Dave DeLong的回答

我有三个具有“长操作”的类,每个类中的操作都没有,或者类之间是非常相关的。一些是网络资源请求,另一些是长处理请求。

- 编辑: 旁注,我似乎有一个问题,我无法为具有多个参数的消息调用运行循环选择器。这是设计限制还是有办法解决这个问题?

例如我有一条消息,例如      - (id)someMessage:(id)value1 otherData:(id)value2 moreData:(id)value3

使runLlector队列runLoop的performSelector函数不支持这样的选择器。

2 个答案:

答案 0 :(得分:5)

我之所以选择longRunningProcess2而不是longRunningProcess3,只是因为如果你能看到协议上的方法声明就更容易理解,而不是依靠文档来计算出回调方法的参数是。

我想补充一点,Apple在10.6新增的API中使用块进行回调,如果你不支持10.5或更早版本,它会为你提供另一种选择。

块方法看起来像这样:

-(void) longRunningProcessWithSuccessHandler:(void(^)(ReturnObject* value))successHandler
                                errorHandler:(void(^)(NSError* error))errorHandler
                             progressHandler:(void(^)(NSInteger completed, NSInteger total))progressHandler;
{
    NSInteger totalItems = 10;
    NSInteger item = 0;
    for(item = 0; item < totalItems; ++item){
        [self processItem:item];
        progressHandler(item, totalItems);
    }

    BOOL wasSuccessful = ?;
    if(wasSuccessful){
        ReturnObject* value = ?;
        successHandler(value);
    } else {
        NSError* error = ?;
        errorHandler(error);
    }
}

你可以这样调用这个方法:

[SomeObj longRunningProcessWithSuccessHandler:^(ReturnObject* value) { [self showReturnObject:value]; }
                                 errorHandler:^(NSError* error){ [self presentError:error]; }
                              progressHandler:^(NSInteger completed, NSInteger total) { [self updateProgressToPercent:(double)completed/total]; }];

答案 1 :(得分:3)

我会使用单一协议路线,类似于您的CallbackProtocolGeneric选项,但我将其扩展为更像:

- (void) operation:(id)operation didFinishWithReturnValue:(id)returnValue;
- (void) operation:(id)operation didFailWithError:(NSError *)error;
- (void) operation:(id)operation hasCompleted:(NSInteger)progress ofTotal:(NSInteger)total;

所以它就像选项1,因为你有一个协议,但是就像选项2一样,你传回了更多的信息。如有必要,您可以使用以下内容进一步扩展:

- (void) operation:(id)operation didFinishStep:(NSInteger)stepNumber withReturnValue:(id)returnValue;
- (void) operation:(id)operation didFailStep:(NSInteger)stepNumber withError:(NSError *)error;
- (void) operation:(id)operation step:(NSInteger)step hasCompleted:(NSInteger)progress ofTotal:(NSInteger)total;

“step”参数可以是一些值,表示该特定对象正在进行的“10+长操作”。

当然,这个建议非常通用,因为你的问题也没有具体的信息,但这可能是我要去的方向(不知道更多)。