我正在为可可应用程序编写中间件,并且正在讨论如何为许多长时间运行的进程设计回调。
当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函数不支持这样的选择器。
答案 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+长操作”。
当然,这个建议非常通用,因为你的问题也没有具体的信息,但这可能是我要去的方向(不知道更多)。