Jak wysłać dane json w żądaniu Http za pomocą NSURLRequest

Jestem nowy w objective-c i zaczynam wkładać wiele wysiłku w prośbę/odpowiedź od niedawna. Mam działający przykład, który może wywołać adres url (przez HTTP GET) i przeanalizować zwrócony json.

Przykład roboczy znajduje się poniżej

- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response {
    [responseData setLength:0];
}

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

- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error {
  NSLog([NSString stringWithFormat:@"Connection failed: %@", [error description]]);
}

- (void)connectionDidFinishLoading:(NSURLConnection *)connection {
    [connection release];
  //do something with the json that comes back ... (the fun part)
}

- (void)viewDidLoad
{
  [self searchForStuff:@"iPhone"];
}

-(void)searchForStuff:(NSString *)text
{
  responseData = [[NSMutableData data] retain];
    NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:@"http://www.whatever.com/json"]];
    [[NSURLConnection alloc] initWithRequest:request delegate:self];
}

Moje pierwsze pytanie brzmi-czy to podejście zwiększy się? Czy to nie jest asynchroniczne (co oznacza, że blokuję wątek interfejsu użytkownika, gdy aplikacja czeka na odpowiedź)

[[3]}moje drugie pytanie brzmi - Jak mogę zmodyfikować część żądania tego zrobić POST zamiast dostać? Czy po prostu modyfikować HttpMethod w ten sposób?
[request setHTTPMethod:@"POST"];

I na koniec - jak dodać zestaw danych json do tego postu jako prosty ciąg znaków (na przykład)

{
    "magic":{
               "real":true
            },
    "options":{
               "happy":true,
                "joy":true,
                "joy2":true
              },
    "key":"123"
}

Z góry dziękuję

Author: vikingosegundo, 2010-12-16

8 answers

Oto co robię (proszę zauważyć, że JSON przechodzący na mój serwer musi być słownikiem z jedną wartością (innym słownikiem) dla key = question..tj. {: pytanie = > {słownik } }):

NSArray *objects = [NSArray arrayWithObjects:[[NSUserDefaults standardUserDefaults]valueForKey:@"StoreNickName"],
  [[UIDevice currentDevice] uniqueIdentifier], [dict objectForKey:@"user_question"],     nil];
NSArray *keys = [NSArray arrayWithObjects:@"nick_name", @"UDID", @"user_question", nil];
NSDictionary *questionDict = [NSDictionary dictionaryWithObjects:objects forKeys:keys];

NSDictionary *jsonDict = [NSDictionary dictionaryWithObject:questionDict forKey:@"question"];

NSString *jsonRequest = [jsonDict JSONRepresentation];

NSLog(@"jsonRequest is %@", jsonRequest);

NSURL *url = [NSURL URLWithString:@"https://xxxxxxx.com/questions"];

NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url
             cachePolicy:NSURLRequestUseProtocolCachePolicy timeoutInterval:60.0];


NSData *requestData = [jsonRequest dataUsingEncoding:NSUTF8StringEncoding];

[request setHTTPMethod:@"POST"];
[request setValue:@"application/json" forHTTPHeaderField:@"Accept"];
[request setValue:@"application/json" forHTTPHeaderField:@"Content-Type"];
[request setValue:[NSString stringWithFormat:@"%d", [requestData length]] forHTTPHeaderField:@"Content-Length"];
[request setHTTPBody: requestData];

NSURLConnection *connection = [[NSURLConnection alloc]initWithRequest:request delegate:self];
if (connection) {
 receivedData = [[NSMutableData data] retain];
}

Otrzymane dane są następnie obsługiwane przez:

NSString *jsonString = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
NSDictionary *jsonDict = [jsonString JSONValue];
NSDictionary *question = [jsonDict objectForKey:@"question"];

To nie jest w 100% Jasne i wymaga ponownego czytania, ale wszystko powinno być tutaj, aby zacząć. I z tego co wiem, to jest asynchroniczne. Mój interfejs nie jest zablokowany podczas wykonywania tych połączeń. Mam nadzieję, że to pomoże.

 102
Author: Mike G,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2013-12-02 09:00:25

Proponuję użyć ASIHTTPRequest

ASIHTTPRequest jest łatwy w użyciu owijanie wokół API CFNetwork, które sprawia, że niektóre z bardziej żmudnych aspektów komunikacji z serwerami WWW łatwiej. Jest napisany w Objective-C i działa zarówno w Mac OS X jak i iPhone aplikacje.

Nadaje się do wykonywania podstawowego HTTP prośby i interakcja z Usługi oparte na odpoczynku (GET / POST / PUT / DELETE). W zestawie Podklasa ASIFormDataRequest sprawia, że łatwe przesyłanie danych i plików postów korzystanie z multipart / form-data.


Należy pamiętać, że oryginalny autor przerwał ten projekt. Zobacz następujący post dla powodów i alternatyw: http://allseeing-i.com/%5Brequest_release%5D ;

Osobiście jestem wielkim fanem AFNetworking

 6
Author: vikingosegundo,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2012-08-01 17:26:32

Walczyłem z tym przez jakiś czas. Uruchamianie PHP na serwerze. Ten kod spowoduje wysłanie json i pobranie odpowiedzi json z serwera

NSURL *url = [NSURL URLWithString:@"http://example.co/index.php"];
NSMutableURLRequest *rq = [NSMutableURLRequest requestWithURL:url];
[rq setHTTPMethod:@"POST"];
NSString *post = [NSString stringWithFormat:@"command1=c1&command2=c2"];
NSData *postData = [post dataUsingEncoding:NSASCIIStringEncoding];
[rq setHTTPBody:postData];
[rq setValue:@"application/x-www-form-urlencoded" forHTTPHeaderField:@"Content-Type"];
NSOperationQueue *queue = [[NSOperationQueue alloc] init];

[NSURLConnection sendAsynchronousRequest:rq queue:queue completionHandler:^(NSURLResponse *response, NSData *data, NSError *error)
 {
     if ([data length] > 0 && error == nil){
         NSError *parseError = nil;
         NSDictionary *dictionary = [NSJSONSerialization JSONObjectWithData:data options:0 error:&parseError];
         NSLog(@"Server Response (we want to see a 200 return code) %@",response);
         NSLog(@"dictionary %@",dictionary);
     }
     else if ([data length] == 0 && error == nil){
         NSLog(@"no data returned");
         //no data, but tried
     }
     else if (error != nil)
     {
         NSLog(@"there was a download error");
         //couldn't download

     }
 }];
 6
Author: user3344717,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2015-02-11 15:03:26

Większość z Was już o tym wie, ale zamieszczam to, ot tak, niektórzy z Was wciąż zmagają się z JSONEM w iOS6+.

W iOS6 i późniejszych mamyklasę NSJSONSerialization , która jest szybka i nie ma zależności od włączania "zewnętrznych" bibliotek.

NSDictionary *result = [NSJSONSerialization JSONObjectWithData:[resultStr dataUsingEncoding:NSUTF8StringEncoding] options:0 error:nil]; 

W ten sposób iOS6 i późniejsze mogą teraz efektywnie parsować JSON.Użycie SBJson jest również implementacją pre-ARC i niesie ze sobą te problemy, jeśli pracujesz w Arc środowisko.

Mam nadzieję, że to pomoże!
 3
Author: tony.stack,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2013-11-12 14:49:47

Oto świetny artykuł przy użyciu Restkit

Wyjaśnia serializację zagnieżdżonych danych do JSON i dołączenie danych do żądania HTTP POST.

 2
Author: cevaris,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2013-08-19 16:34:57

Ponieważ moja edycja odpowiedzi Mike ' a G na modernizację kodu została odrzucona 3 do 2 jako

Ta edycja miała na celu zwrócenie się do autora postu i nie sens jako edycja. Powinien być napisany jako komentarz lub odpowiedź

Przerzucam swoją edycję jako osobną odpowiedź tutaj. Ta edycja usuwa zależność JSONRepresentation z {[3] } Jak sugeruje komentarz Roba z 15 głosami.

    NSArray *objects = [NSArray arrayWithObjects:[[NSUserDefaults standardUserDefaults]valueForKey:@"StoreNickName"],
      [[UIDevice currentDevice] uniqueIdentifier], [dict objectForKey:@"user_question"],     nil];
    NSArray *keys = [NSArray arrayWithObjects:@"nick_name", @"UDID", @"user_question", nil];
    NSDictionary *questionDict = [NSDictionary dictionaryWithObjects:objects forKeys:keys];

    NSDictionary *jsonDict = [NSDictionary dictionaryWithObject:questionDict forKey:@"question"];

    NSLog(@"jsonRequest is %@", jsonRequest);

    NSURL *url = [NSURL URLWithString:@"https://xxxxxxx.com/questions"];

    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url
                 cachePolicy:NSURLRequestUseProtocolCachePolicy timeoutInterval:60.0];


    NSData *requestData = [NSJSONSerialization dataWithJSONObject:dict options:0 error:nil]; //TODO handle error

    [request setHTTPMethod:@"POST"];
    [request setValue:@"application/json" forHTTPHeaderField:@"Accept"];
    [request setValue:@"application/json" forHTTPHeaderField:@"Content-Type"];
    [request setValue:[NSString stringWithFormat:@"%d", [requestData length]] forHTTPHeaderField:@"Content-Length"];
    [request setHTTPBody: requestData];

    NSURLConnection *connection = [[NSURLConnection alloc]initWithRequest:request delegate:self];
    if (connection) {
     receivedData = [[NSMutableData data] retain];
    }

Otrzymane dane są następnie obsługiwane przez:

NSDictionary *jsonDict = [NSJSONSerialization JSONObjectWithData:data options:0 error:nil];
    NSDictionary *question = [jsonDict objectForKey:@"question"];
 2
Author: Steve Moser,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2015-11-03 17:54:46

Oto zaktualizowany przykład, który używa NSURLConnection +sendAsynchronousRequest: (10.7+, iOS 5+), żądanie "Post" pozostaje takie samo jak w przypadku zaakceptowanej odpowiedzi i jest tutaj pominięte dla jasności:

NSURL *apiURL = [NSURL URLWithString:
    [NSString stringWithFormat:@"http://www.myserver.com/api/api.php?request=%@", @"someRequest"]];
NSURLRequest *request = [NSURLRequest requestWithURL:apiURL]; // this is using GET, for POST examples see the other answers here on this page
[NSURLConnection sendAsynchronousRequest:request
                                   queue:[NSOperationQueue mainQueue]
                       completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
     if(data.length) {
         NSString *responseString = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
         if(responseString && responseString.length) {
             NSLog(@"%@", responseString);
         }
     }
}];
 0
Author: auco,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2013-12-09 14:42:42

Możesz wypróbować ten kod do wysłania ciągu json

NSData *jsonData = [NSJSONSerialization dataWithJSONObject:ARRAY_CONTAIN_JSON_STRING options:NSJSONWritin*emphasized text*gPrettyPrinted error:NULL];
NSString *jsonString = [[NSString alloc] initWithData:jsonData encoding:NSUTF8StringEncoding];
NSString *WS_test = [NSString stringWithFormat:@"www.test.com?xyz.php&param=%@",jsonString];
 0
Author: jayesh mardiya,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2016-10-04 12:28:52