什么是 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) 变得没有响应。在异步操作中,您创建了另一条路(读作:线程),行驶缓慢的汽车可以切换到那条路。一旦它的速度足以在主干道上运行,它就可以切换回主干道。

在回答您的问题时,同步请求会阻塞调用它们的线程,直到请求完成。 (因此,通常不鼓励同步请求。)异步请求让当前线程在执行请求时继续执行(例如,继续响应用户与应用程序的交互;响应系统事件等)。这通常更可取。

您的代码正在执行异步请求(这很好)。不过还是有一些问题:

  1. 你不应该再使用 NSURLConnection。它已被弃用。使用 NSURLSession。它实际上更简单,因为您通常不必编写那些委托方法(除非您想要这样做,因为您有一些迫切需要这样做)。

    有关 NSURLSession 的更多信息,请参阅 Using NSURLSession in the URL Session Programming Guide. Or see WWDC 2013 video What's New in Foundation Networking 以获得很好的介绍。

  2. 您没有进行错误处理。您正在检查基本错误(例如没有网络),这非常好,但您没有考虑其他网络服务器错误,这些错误可能并不总是导致 NSError 对象,而可能只是导致 HTTP 状态200 以外的代码。我建议检查一下。

    有关 HTTP 状态代码列表,请参阅 RFC 2616 的第 10 部分。

  3. 您正在设置 application/jsonContent-Type。但这不是 JSON 请求。 (当然,响应是 JSON,但请求不是。)通常您会使用 application/x-www-form-urlencoded 来处理这样的请求。

  4. 在您的代码片段中,您建议来自服务器的响应是 JSON,其中 NSArray 作为顶级对象。但是顶级对象是 NSDictionary.

  5. 您正在使用 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.