Как отправить данные JSON в запрос Http с помощью NSURLRequest - PullRequest
78 голосов
/ 16 декабря 2010

Я новичок в объективе-c, и в последнее время я начинаю прилагать большие усилия к запросу / ответу. У меня есть рабочий пример, который может вызвать URL (через http GET) и проанализировать возвращенный JSON.

Рабочий пример этого ниже

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

Мой первый вопрос - будет ли этот подход расширяться? Или это не асинхронно (то есть я блокирую поток пользовательского интерфейса, пока приложение ожидает ответа)

Мой второй вопрос - как я могу изменить часть запроса, чтобы сделать POST вместо GET? Это просто изменить HttpMethod, как это?

[request setHTTPMethod:@"POST"];

И, наконец, - как мне добавить в этот пост набор данных json в виде простой строки (например)

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

Заранее спасибо

Ответы [ 8 ]

103 голосов
/ 17 декабря 2010

Вот что я делаю (обратите внимание, что JSON, идущий на мой сервер, должен быть словарем с одним значением (другим словарем) для 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];
}

Полученные данные затем обрабатываются:

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

Это не на 100% ясно и потребует некоторого перечитывания, но все должно быть здесь, чтобы вы начали.И из того, что я могу сказать, это асинхронный.Мой пользовательский интерфейс не заблокирован во время этих звонков.Надеюсь, это поможет.

7 голосов
/ 11 февраля 2015

Я боролся с этим некоторое время. Запуск PHP на сервере. Этот код отправит json и получит ответ json от сервера

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 голосов
/ 16 декабря 2010

Я бы предложил использовать ASIHTTPRequest

ASIHTTPRequest - это простая в использовании оболочка для API-интерфейса CFNetwork, упрощающая некоторые из утомительных аспектов взаимодействия с веб-серверами.Он написан на Objective-C и работает в приложениях Mac OS X и iPhone.

Подходит для выполнения базовых HTTP-запросов и взаимодействия со службами на основе REST (GET / POST / PUT / DELETE).Включенный подкласс ASIFormDataRequest позволяет легко отправлять данные и файлы POST, используя multipart / form-data.


Обратите внимание, что первоначальный автор прекратил работу над этим проектом.Для объяснения причин и альтернатив см. Следующий пост: http://allseeing -i.com /% 5Brequest_release% 5D ;

Лично я большой поклонник AFNetworking

3 голосов
/ 12 ноября 2013

Большинство из вас уже знают это, но я публикую это, только что некоторые из вас все еще борются с JSON в iOS6 +.

В iOS6 и более поздних версиях у нас есть NSJSONSerializationКласс , который быстр и не зависит от включения "внешних" библиотек.

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

Это способ, которым iOS6 и более поздние версии теперь могут эффективно анализировать JSON. Использование SBJson также является реализацией до ARCи приносит с собой и эти проблемы, если вы работаете в среде ARC.

Надеюсь, это поможет!

2 голосов
/ 03 ноября 2015

С тех пор, как мое изменение в ответе Майка Дж на модернизацию кода было отклонено 3 к 2 как

Это изменение предназначалось для автора сообщения и не имеет смысла как редактирование.Это должно было быть написано как комментарий или ответ

Я отправляю свою правку как отдельный ответ здесь.Это редактирование удаляет зависимость JSONRepresentation с NSJSONSerialization, как предполагает комментарий Роба с 15 ответами.

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

Полученные данные затем обрабатываются:

NSDictionary *jsonDict = [NSJSONSerialization JSONObjectWithData:data options:0 error:nil];
    NSDictionary *question = [jsonDict objectForKey:@"question"];
2 голосов
/ 19 августа 2013

Вот отличная статья с использованием Restkit

Он объясняет сериализацию вложенных данных в JSON и присоединение данных к HTTP-запросу POST.

0 голосов
/ 04 октября 2016

Вы можете попробовать этот код для отправки строки 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 голосов
/ 09 декабря 2013

Вот обновленный пример, который использует NSURLConnection + sendAsynchronousRequest: (10.7+, iOS 5+). Запрос «Post» остается таким же, как и с принятым ответом, и здесь для ясности опущен:

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