基本上我得到了 2 个 NSURLConnections。 第一个从我的 API 获取 token ,第二个从 API 获取一些事件,但需要 token 才能执行此操作。我目前正在学习 Objective-C/Ios 的东西,所以这不是一个真正的应用程序。 我要做的是通过我的类属性 current_connection 区分这两个连接。
1) 我的第一个问题是:有没有更好的方法来做到这一点?
2) 第二个问题:像这样建立两个连接安全吗?因为这些必须按这个顺序发生,我不知道如果我的代码是这样的话IOS是否会一直这样。例如,如果第一个请求需要很长时间,它是否已经开始第二个?如果没有 token ,那将是一个问题。
3) 我读到了 https://github.com/AFNetworking/AFNetworking - 我应该使用它吗?
我的代码:
HomeViewController.h
#import <UIKit/UIKit.h>
@interface HomeViewController : UIViewController
@property (nonatomic, strong) NSString* current_connection;
@end
我的 HomeViewController.m
#import "HomeViewController.h"
@interface HomeViewController ()
@end
@implementation HomeViewController
@synthesize current_connection;
- (void)viewDidLoad {
[super viewDidLoad];
//get user token
NSString *post = [NSString stringWithFormat:@"&email=%@&password=%@",[[NSUserDefaults standardUserDefaults] objectForKey:@"email"], [[NSUserDefaults standardUserDefaults] objectForKey:@"password"]];
NSData *postData = [post dataUsingEncoding:NSASCIIStringEncoding allowLossyConversion:YES];
NSString *postLength = [NSString stringWithFormat:@"%lu",(unsigned long)[postData length]];
NSMutableURLRequest *request = [[NSMutableURLRequest alloc] init];
[request setURL:[NSURL URLWithString:[NSString stringWithFormat:@"http://localhost:3000/api/v1/sessions"]]];
[request setHTTPMethod:@"POST"];
[request setValue:postLength forHTTPHeaderField:@"Content-Length"];
[request setValue:@"application/x-www-form-urlencoded" forHTTPHeaderField:@"Current-Type"];
[request setHTTPBody:postData];
current_connection = @"token";
NSURLConnection *conn = [[NSURLConnection alloc]initWithRequest:request delegate:self];
//load events
NSUserDefaults* defaults = [NSUserDefaults standardUserDefaults];
NSString *url =[NSString stringWithFormat:@"http://localhost:3000/api/v1/events/start_events?user_email=%@&user_token=%@",[[NSUserDefaults standardUserDefaults] objectForKey:@"email"], [[NSUserDefaults standardUserDefaults] objectForKey:@"token"]];
NSMutableURLRequest *req = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:url]];
[req setHTTPMethod:@"GET"];
current_connection = @"event";
NSURLConnection *connection = [[NSURLConnection alloc] initWithRequest:req delegate:self];
}
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData*)data {
NSError *error;
NSMutableDictionary *cdata = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableContainers|NSJSONReadingMutableLeaves error:&error];
if( error )
{
NSLog(@"%@", [error localizedDescription]);
}
else {
if ([current_connection isEqual: @"token"]) {
NSString *token = cdata[@"token"];
NSLog(@"%@", token);
//save token to user defaults
NSUserDefaults* defaults = [NSUserDefaults standardUserDefaults];
[defaults setObject:token forKey:@"token"];
[defaults synchronize];
}
else {
//display events
NSLog(@"%@", cdata[@"events"]);
}
}
}
@end
最佳答案
回答您的问题:
是的,有更好的方法可以做到这一点。有很多问题:
您的两个连接请求将调用同一个委托(delegate)对象,除非您非常小心,否则它们将相互跨越。
在您的情况下,由于第二个连接需要第一个连接完成(即您需要 token ,大概是在开始下一个连接之前),您可以通过不立即启动第二个连接来解决此问题在第一个启动之后,但在第一个完成的地方(即您获得 token 的地方)。
现在,您的两个请求实际上同时运行(您无法正式保证哪个将先完成)。您想要更改此设置以便它们连续运行。
您正在处理
didReceiveData
中的响应。这很危险,因为有时响应正文需要多次调用didReceiveData
。通常你 (a) 在发起请求之前实例化一些NSMutableData
对象; (b)didReceiveData
只是将接收到的数据附加到NSMutableData
中; (c) 仅在connectionDidFinishLoading
中,您现在会尝试使用整个NSMutableData
,并处理它以获取 Web 服务的响应。
最重要的是,这个基于委托(delegate)的实现有很多问题,我建议使用
sendAsynchronousRequest
(一个很好的基于 block 的实现,让你摆脱这些杂草)或 AFNetworking(一个很好的在NSOperation
对象中包装基于委托(delegate)的实现)。您可以修复此基于委托(delegate)的实现,但使用上述方法之一可能更容易。不,使用同一个委托(delegate)对象发出两个并发请求是不明智的。现在,事实证明你可能不希望它们是并发的,无论如何,但作为一般设计原则,如果你为每个
NSURLConnection
请求有不同的委托(delegate)对象,生活会更容易。是的,AFNetworking 可能值得一试。它并不完美,但它可以让您摆脱
NSURLConnectionDataDelegate
编程的困扰。如果您不需要 AFNetworking 的全部丰富性,您也可以使用NSURLConnection
类方法sendAsynchronousRequest
发出像这些一样简单的请求。
关于ios - 连续多个 NSURLConnection,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26361279/