gpt4 book ai didi

ios - 是否有比 Objective-C 中的委托(delegate)更简洁的方法来处理异步 http 请求?

转载 作者:行者123 更新时间:2023-11-28 21:58:46 26 4
gpt4 key购买 nike

我正在构建一个 iPhone 应用程序,它大量利用了我用 php 编写的网络服务。现在我有一个文件 (API.h/API.m),其中包含通过 NSURLConnection 在我的 Web 服务上调用 php 函数的方法。

方法如:

-(void) mediaAdd:(UIImage *)image withDelegate: (id<apiRequest>) delegate;
-(void) mediaGet:(NSString *)imageURL withDelegate: (id<apiRequest>) delegate;

这些方法由应用程序中的 Controller (MVC 模式定义的 Controller )调用,这些方法将 Controller 作为委托(delegate)。

一旦 API.h/API.m 中被调用方法的请求完成,API.h/API.m 中的 NSURLConnection 委托(delegate)方法在请求完成时调用,然后调用委托(delegate)方法...

-(void) serverResponse:(NSDictionary *) response fromCall:(NSString *)call;

...这当然是在调用 API.h/API.m 中的方法的 Controller 上执行的。

当 Controller 需要进行多个 API 调用时,就会出现我的问题,或者更确切地说是困惑。因为我只有一个委托(delegate)方法,所以我通过在 serverResponse 方法中使用调用参数来区分不同的调用。

这是 serverResponse 方法实例的框架:

//API Delegate
-(void)serverResponse:(NSDictionary *)response fromCall:(NSString *)call {
bool local = false;
NSString *callbackString;
if (local) {
callbackString = @"http://localhost/";
} else {
callbackString = @"http://secret.com/";
}
if ([call isEqualToString:[NSString stringWithFormat:@"%@user/add",callbackString]]) {
//user/add was called
if (![[response objectForKey:@"status"] isEqualToString:@"fail"]) {
if ([[response objectForKey:@"status"] isEqualToString:@"fail"]) {
if ([[response objectForKey:@"reason"] isEqualToString:@"udid already taken"]) {
//UDID taken
}
} else {

//UDID not taken, we're good to go



}

} else {
NSLog(@"fail");
}
} else if([call isEqualToString:[NSString stringWithFormat:@"%@moment/get",callbackString]]){
//moment/get was the api call
} else if([call isEqualToString:[NSString stringWithFormat:@"%@printPictures/printPic",callbackString]]){

//printPictures/printPic was the api call
} else {
NSLog(@"wrong call");
}



}

我的问题是,我是否应该继续为每个 API 调用创建一个委托(delegate)方法?还是我想太多了,实际上有一个非常简单的方法来处理这个问题。也许是设计模式或结构?

谢谢!-迈克

最佳答案

我通过一个带有回调 block 的单独下载类来做这种事情。 Controller 创建该类的一个实例,并在回调中接收结果,之后释放下载类。您需要进行的每个调用都会创建自己的 Downloader 类实例,因此您无需跟踪哪个响应与哪个调用相关。

    Downloader *dl = [Downloader new];
NSURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:[NSString stringWithFormat:@"...."]] cachePolicy:NSURLRequestReloadIgnoringLocalAndRemoteCacheData timeoutInterval:10];

[dl downloadWithRequest:(NSURLRequest *) request completion:^(NSData *data, NSError *error) {
if (!error ) {
// do whatever you need to do with the raw data here
NSLog(@"got the data");
}else{
NSLog(@"%@",error);
}
}];

下面是我如何实现 Downloader 类。

typedef void(^compBlock)(NSData *data, NSError *error);

@interface Downloader : NSObject <NSURLConnectionDataDelegate>

-(void)downloadWithRequest:(NSURLRequest *) request completion:(compBlock) completion;

@end

实现文件,

@interface Downloader()
@property (copy,nonatomic) compBlock completionHandler;
@property (strong,nonatomic) NSMutableData *receivedData;
@end

@implementation Downloader

-(void)downloadWithRequest:(NSURLRequest *) request completion:(compBlock)completion {
NSURLConnection *con = [NSURLConnection connectionWithRequest:request delegate:self];
if (con) {
self.receivedData = [NSMutableData new];
self.completionHandler = completion;
}
}


-(void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response {
self.receivedData.length = 0;
}



-(void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data {
[self.receivedData appendData:data];
}



-(void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error {
self.completionHandler(nil, error);
}



-(void)connectionDidFinishLoading:(NSURLConnection *)connection {
self.completionHandler(self.receivedData, nil);
}

关于ios - 是否有比 Objective-C 中的委托(delegate)更简洁的方法来处理异步 http 请求?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25669761/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com