什么是 diff b/w 同步和异步请求,如何以编程方式检查此操作
what is diff b/w Synchronous and asynchronous requests ,How check this operations programmatically
嗨,我是 ios 的初学者,当我们使用 NSURLRequest 调用服务时,我想知道当我们使用 "Synchronous request" 调用服务和使用异步请求调用服务时会发生什么以编程方式,
请以编程方式解释操作,我在下面使用该代码编写了一些代码来解释同步和异步操作
我的代码:-
- (void)viewDidLoad {
[super viewDidLoad];
NSURL *url = [NSURL URLWithString:@"http://api.kivaws.org/v1/loans/search.json?status=fundraising"];
NSMutableURLRequest *theRequest = [NSMutableURLRequest requestWithURL:url];
[theRequest setHTTPMethod:@"GET"];
[theRequest setValue:@"application/json" forHTTPHeaderField:@"Accept"];
[theRequest setValue:@"application/json" forHTTPHeaderField:@"Content-Type"];
[theRequest setTimeoutInterval:5];
NSURLConnection *connection = [[NSURLConnection alloc] initWithRequest:theRequest delegate:self];
if(connection){
webData = [[NSMutableData alloc] init];
}
}
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response {
[webData setLength:0];
}
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data {
[webData appendData:data];
}
- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error {
NSLog(@"error is %@",[error localizedDescription]);
}
- (void)connectionDidFinishLoading:(NSURLConnection *)connection {
NSString * allDataDictionbary = [[NSString alloc] initWithData:webData encoding:NSUTF8StringEncoding];
NSArray * responseString = [allDataDictionbary JSONValue];
NSLog(@"final respone dictionary%@",responseString);
}
同步请求阻塞线程直到它完成。而异步请求创建单独的线程并执行,完成后返回主线程。
当你运行这条线有1秒的时差或取决于网速
NSLog(@"before SynchronousRequest time %@",[NSDate date]);
NSError *error = nil;
NSHTTPURLResponse *response = nil;
NSData *data=[NSURLConnection sendSynchronousRequest:request returningResponse:&response error:&error];
NSLog(@"after SynchronousRequest time %@",[NSDate date]);
SynchronousRequest时间2015-12-12之前09:26:01+0000
SynchronousRequest 时间 2015-12-12 之后 09:26:02 +0000
当你运行这条线没有时差
NSLog(@"before AsynchronousRequest time %@",[NSDate date]);
[NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *error) {
if (data) {
NSString *str = [[NSString alloc] initWithData:data
encoding:NSUTF8StringEncoding]; // note the retain count here.
NSLog(@"%@",str);
} else {
// handle error
}
}];
NSLog(@"after AsynchronousRequest time %@",[NSDate date]);
AsynchronousRequest时间2015-12-12之前09:29:50+0000
AsynchronousRequest 时间 2015-12-12 之后 09:29:50 +0000
当从一个线程发起 HTTP 请求时,如果您不介意耐心等待 ("thank you for waiting"),该线程可以决定自行处理所有响应。该线程被阻塞,直到 HTTP 请求被完全处理。或者,线程可能认为他有更好的事情要做,而不是等待冗长的 HTTP 事务,因此他愉快地将任务传递给另一个进程来处理 HTTP 事务,以及一个回调,以便他们保持联系。当另一个进程完成时,它使用回调函数返回它的发起者它已经完成。
把它想象成一条路。想象一下,您的应用程序将只在一条道路上运行,如果没有被告知要这样做的话。现在,您的应用程序中发生的一切都必须保持在这条路上。如果任何 "car"(read: task) 花费的时间太长,整个事情就会关闭并且您的 "road"(read: app) 变得没有响应。在异步操作中,您创建了另一条路(读作:线程),行驶缓慢的汽车可以切换到那条路。一旦它的速度足以在主干道上运行,它就可以切换回主干道。
在回答您的问题时,同步请求会阻塞调用它们的线程,直到请求完成。 (因此,通常不鼓励同步请求。)异步请求让当前线程在执行请求时继续执行(例如,继续响应用户与应用程序的交互;响应系统事件等)。这通常更可取。
您的代码正在执行异步请求(这很好)。不过还是有一些问题:
你不应该再使用 NSURLConnection
。它已被弃用。使用 NSURLSession
。它实际上更简单,因为您通常不必编写那些委托方法(除非您想要这样做,因为您有一些迫切需要这样做)。
有关 NSURLSession
的更多信息,请参阅 Using NSURLSession in the URL Session Programming Guide. Or see WWDC 2013 video What's New in Foundation Networking 以获得很好的介绍。
您没有进行错误处理。您正在检查基本错误(例如没有网络),这非常好,但您没有考虑其他网络服务器错误,这些错误可能并不总是导致 NSError
对象,而可能只是导致 HTTP 状态200 以外的代码。我建议检查一下。
有关 HTTP 状态代码列表,请参阅 RFC 2616 的第 10 部分。
您正在设置 application/json
的 Content-Type
。但这不是 JSON 请求。 (当然,响应是 JSON,但请求不是。)通常您会使用 application/x-www-form-urlencoded
来处理这样的请求。
在您的代码片段中,您建议来自服务器的响应是 JSON,其中 NSArray
作为顶级对象。但是顶级对象是 NSDictionary
.
您正在使用 JSONValue
。我不确定那是哪个 JSON 库,但我们中的许多人只是使用 Apple 提供的内置 NSJSONSerialization
class。很久以前,在 Apple 提供 NSJSONSerialization
之前,我们会使用第三方库来解析 JSON,但现在不再需要了。
使用NSURLSession
发送请求的正确方式如下:
NSURL *url = [NSURL URLWithString:@"http://api.kivaws.org/v1/loans/search.json?status=fundraising"];
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
[request setHTTPMethod:@"GET"];
[request setValue:@"application/json" forHTTPHeaderField:@"Accept"];
[request setValue:@"application/x-www-form-urlencoded" forHTTPHeaderField:@"Content-Type"];
NSURLSessionTask *task = [[NSURLSession sharedSession] dataTaskWithRequest:request completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {
if (error != nil) {
NSLog(@"fundamental network error = %@", error);
return;
}
if ([response isKindOfClass:[NSHTTPURLResponse class]]) {
NSInteger statusCode = [(NSHTTPURLResponse *)response statusCode];
if (statusCode != 200) {
NSLog(@"Warning; server should respond with 200 status code, but returned %ld", (long)statusCode);
}
}
NSError *parseError;
NSDictionary *responseObject = [NSJSONSerialization JSONObjectWithData:data options:0 error:&parseError];
if (responseObject) {
NSLog(@"responseObject = %@", responseObject);
} else {
NSLog(@"Error parsing JSON: %@", parseError);
NSString *responseString = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
NSLog(@"responseString = %@", responseString);
}
}];
[task resume];
// Note, you'll reach this portion of your code before the
// above `completionHandler` runs because this request runs
// asynchronously. So put code that uses the network response
// above, inside that `completionHandler`, not here.
嗨,我是 ios 的初学者,当我们使用 NSURLRequest 调用服务时,我想知道当我们使用 "Synchronous request" 调用服务和使用异步请求调用服务时会发生什么以编程方式, 请以编程方式解释操作,我在下面使用该代码编写了一些代码来解释同步和异步操作
我的代码:-
- (void)viewDidLoad {
[super viewDidLoad];
NSURL *url = [NSURL URLWithString:@"http://api.kivaws.org/v1/loans/search.json?status=fundraising"];
NSMutableURLRequest *theRequest = [NSMutableURLRequest requestWithURL:url];
[theRequest setHTTPMethod:@"GET"];
[theRequest setValue:@"application/json" forHTTPHeaderField:@"Accept"];
[theRequest setValue:@"application/json" forHTTPHeaderField:@"Content-Type"];
[theRequest setTimeoutInterval:5];
NSURLConnection *connection = [[NSURLConnection alloc] initWithRequest:theRequest delegate:self];
if(connection){
webData = [[NSMutableData alloc] init];
}
}
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response {
[webData setLength:0];
}
- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data {
[webData appendData:data];
}
- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error {
NSLog(@"error is %@",[error localizedDescription]);
}
- (void)connectionDidFinishLoading:(NSURLConnection *)connection {
NSString * allDataDictionbary = [[NSString alloc] initWithData:webData encoding:NSUTF8StringEncoding];
NSArray * responseString = [allDataDictionbary JSONValue];
NSLog(@"final respone dictionary%@",responseString);
}
同步请求阻塞线程直到它完成。而异步请求创建单独的线程并执行,完成后返回主线程。
当你运行这条线有1秒的时差或取决于网速
NSLog(@"before SynchronousRequest time %@",[NSDate date]);
NSError *error = nil;
NSHTTPURLResponse *response = nil;
NSData *data=[NSURLConnection sendSynchronousRequest:request returningResponse:&response error:&error];
NSLog(@"after SynchronousRequest time %@",[NSDate date]);
SynchronousRequest时间2015-12-12之前09:26:01+0000
SynchronousRequest 时间 2015-12-12 之后 09:26:02 +0000
当你运行这条线没有时差
NSLog(@"before AsynchronousRequest time %@",[NSDate date]);
[NSURLConnection sendAsynchronousRequest:request queue:[NSOperationQueue mainQueue] completionHandler:^(NSURLResponse *response, NSData *data, NSError *error) {
if (data) {
NSString *str = [[NSString alloc] initWithData:data
encoding:NSUTF8StringEncoding]; // note the retain count here.
NSLog(@"%@",str);
} else {
// handle error
}
}];
NSLog(@"after AsynchronousRequest time %@",[NSDate date]);
AsynchronousRequest时间2015-12-12之前09:29:50+0000
AsynchronousRequest 时间 2015-12-12 之后 09:29:50 +0000
当从一个线程发起 HTTP 请求时,如果您不介意耐心等待 ("thank you for waiting"),该线程可以决定自行处理所有响应。该线程被阻塞,直到 HTTP 请求被完全处理。或者,线程可能认为他有更好的事情要做,而不是等待冗长的 HTTP 事务,因此他愉快地将任务传递给另一个进程来处理 HTTP 事务,以及一个回调,以便他们保持联系。当另一个进程完成时,它使用回调函数返回它的发起者它已经完成。
把它想象成一条路。想象一下,您的应用程序将只在一条道路上运行,如果没有被告知要这样做的话。现在,您的应用程序中发生的一切都必须保持在这条路上。如果任何 "car"(read: task) 花费的时间太长,整个事情就会关闭并且您的 "road"(read: app) 变得没有响应。在异步操作中,您创建了另一条路(读作:线程),行驶缓慢的汽车可以切换到那条路。一旦它的速度足以在主干道上运行,它就可以切换回主干道。
在回答您的问题时,同步请求会阻塞调用它们的线程,直到请求完成。 (因此,通常不鼓励同步请求。)异步请求让当前线程在执行请求时继续执行(例如,继续响应用户与应用程序的交互;响应系统事件等)。这通常更可取。
您的代码正在执行异步请求(这很好)。不过还是有一些问题:
你不应该再使用
NSURLConnection
。它已被弃用。使用NSURLSession
。它实际上更简单,因为您通常不必编写那些委托方法(除非您想要这样做,因为您有一些迫切需要这样做)。有关
NSURLSession
的更多信息,请参阅 Using NSURLSession in the URL Session Programming Guide. Or see WWDC 2013 video What's New in Foundation Networking 以获得很好的介绍。您没有进行错误处理。您正在检查基本错误(例如没有网络),这非常好,但您没有考虑其他网络服务器错误,这些错误可能并不总是导致
NSError
对象,而可能只是导致 HTTP 状态200 以外的代码。我建议检查一下。有关 HTTP 状态代码列表,请参阅 RFC 2616 的第 10 部分。
您正在设置
application/json
的Content-Type
。但这不是 JSON 请求。 (当然,响应是 JSON,但请求不是。)通常您会使用application/x-www-form-urlencoded
来处理这样的请求。在您的代码片段中,您建议来自服务器的响应是 JSON,其中
NSArray
作为顶级对象。但是顶级对象是NSDictionary
.您正在使用
JSONValue
。我不确定那是哪个 JSON 库,但我们中的许多人只是使用 Apple 提供的内置NSJSONSerialization
class。很久以前,在 Apple 提供NSJSONSerialization
之前,我们会使用第三方库来解析 JSON,但现在不再需要了。
使用NSURLSession
发送请求的正确方式如下:
NSURL *url = [NSURL URLWithString:@"http://api.kivaws.org/v1/loans/search.json?status=fundraising"];
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url];
[request setHTTPMethod:@"GET"];
[request setValue:@"application/json" forHTTPHeaderField:@"Accept"];
[request setValue:@"application/x-www-form-urlencoded" forHTTPHeaderField:@"Content-Type"];
NSURLSessionTask *task = [[NSURLSession sharedSession] dataTaskWithRequest:request completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {
if (error != nil) {
NSLog(@"fundamental network error = %@", error);
return;
}
if ([response isKindOfClass:[NSHTTPURLResponse class]]) {
NSInteger statusCode = [(NSHTTPURLResponse *)response statusCode];
if (statusCode != 200) {
NSLog(@"Warning; server should respond with 200 status code, but returned %ld", (long)statusCode);
}
}
NSError *parseError;
NSDictionary *responseObject = [NSJSONSerialization JSONObjectWithData:data options:0 error:&parseError];
if (responseObject) {
NSLog(@"responseObject = %@", responseObject);
} else {
NSLog(@"Error parsing JSON: %@", parseError);
NSString *responseString = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
NSLog(@"responseString = %@", responseString);
}
}];
[task resume];
// Note, you'll reach this portion of your code before the
// above `completionHandler` runs because this request runs
// asynchronously. So put code that uses the network response
// above, inside that `completionHandler`, not here.