Como enviar dados json na solicitação Http usando NSURLRequest

Eu sou novo no objective-c e estou começando a fazer um grande esforço em solicitação / resposta como de recente. Eu tenho um exemplo de trabalho que pode chamar um url (via http GET) e analisar o json retornado.

O exemplo prático disso está abaixo

- (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]; } 

Minha primeira pergunta é: essa abordagem será ampliada? Ou isso não é asynchronous (o que significa bloquear o thread da interface do usuário enquanto o aplicativo está aguardando a resposta)

Minha segunda pergunta é: como modificar a parte da solicitação para fazer um POST em vez de GET? É simplesmente para modificar o HttpMethod como assim?

 [request setHTTPMethod:@"POST"]; 

E finalmente – como eu adiciono um dataset json a este post como uma string simples (por exemplo)

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

Agradeço antecipadamente

Aqui está o que eu faço (por favor, note que o JSON indo para o meu servidor precisa ser um dictionary com um valor (outro dictionary) para key = question..ie {: question => {dictionary}}):

 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]; } 

O receivedData é então manipulado por:

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

Isso não é 100% claro e levará algumas releituras, mas tudo deve estar aqui para você começar. E pelo que posso dizer, isso é asynchronous. Minha interface do usuário não está bloqueada enquanto essas chamadas são feitas. Espero que ajude.

Eu sugiro usar o ASIHTTPRequest

O ASIHTTPRequest é um wrapper fácil de usar em torno da API do CFNetwork que facilita alguns dos aspectos mais tediosos da comunicação com servidores da web. Está escrito em Objective-C e funciona em aplicativos Mac OS X e iPhone.

É adequado executar solicitações HTTP básicas e interagir com serviços baseados em REST (GET / POST / PUT / DELETE). A subclass ASIFormDataRequest incluída facilita o envio de dados e arquivos POST usando multipart / form-data.


Por favor, note que o autor original interrompeu com este projeto. Veja a postagem seguinte por razões e alternativas: http://allseeing-i.com/%5Brequest_release%5D ;

Pessoalmente eu sou um grande fã da AFNetworking

Eu lutei com isso por um tempo. Executando PHP no servidor. Este código irá postar um json e obter a resposta json do servidor

 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 } }]; 

A maioria de vocês já sabe disso agora, mas estou postando isso, apenas no caso de alguns de vocês ainda estão lutando com o JSON no iOS6 +.

No iOS6 e posterior, temos a NSJSONSerialization Class que é rápida e não depende de include bibliotecas “externas”.

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

Essa é a maneira como o iOS6 e mais recente podem agora analisar o JSON com eficiência. O uso do SBJson também é uma implementação pré-ARC e traz consigo esses problemas também se você estiver trabalhando em um ambiente ARC.

Eu espero que isso ajude!

Aqui está um ótimo artigo usando o Restkit

Explica na serialização de dados nesteds em JSON e anexando os dados a uma solicitação HTTP POST.

Desde a minha edição para a resposta de Mike G para modernizar o código foi rejeitada 3 a 2 como

Esta edição pretendia abordar o autor do post e não faz sentido como edição. Deveria ter sido escrito como um comentário ou uma resposta

Estou repostando minha edição como uma resposta separada aqui. Esta edição remove a dependência JSONRepresentation com NSJSONSerialization como NSJSONSerialization o comentário de Rob com 15 NSJSONSerialization .

  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]; } 

O receivedData é então manipulado por:

 NSDictionary *jsonDict = [NSJSONSerialization JSONObjectWithData:data options:0 error:nil]; NSDictionary *question = [jsonDict objectForKey:@"question"]; 

Aqui está um exemplo atualizado que está usando NSURLConnection + sendAsynchronousRequest: (10.7+, iOS 5+), A solicitação “Post” permanece a mesma que com a resposta aceita e é omitida aqui para fins de clareza:

 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); } } }]; 

Você pode tentar este código para enviar json string

 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];