Як надіслати дані JSON у запиті Http за допомогою NSURLRequest


80

Я новачок у task-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"
}

Заздалегідь спасибі


Відповіді:


105

Ось що я роблю (зауважте, що JSON, який переходить на мій сервер, повинен бути словником з одним значенням (іншим словником) для ключа = питання..і {: питання => {словник}}):

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% зрозуміло, і потрібно буде перечитати, але все має бути тут, щоб ви почали. І з того, що я можу сказати, це асинхронно. Мій інтерфейс не блокується під час здійснення цих дзвінків. Сподіваюся, що це допомагає.


Все виглядає добре, крім цього рядка [dict objectForKey: @ "user_question"], nil]; - dict не задекларований у вашій вибірці. Це просто простий словник чи щось особливе?
Торан Біллапс

1
Вибач за це. Так, "dict" - це просто простий словник, який я завантажую з документів користувачів iOS.
Mike G

19
Це використовується NSDictionaryметод екземпляра JSONRepresentation. Я можу запропонувати використовувати NSJSONSerializationметод класу dataWithJSONObjectзамість json-framework .
Роб

Більш ефективно перетворити NSUInteger на NSString через NSNumber like [[NSNumber numberWithUnsignedInt:requestData.length] stringValue].
respectTheCode

1
@MikeG Виправлено давню і поки непомічену помилку у зразку коду. Вибачте, за редагування вашого допису;)
CouchDeveloper

7

Я боровся з цим деякий час. Запуск 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

     }
 }];

1
content type = "application / x-www-form-urlencoded" зробив свою справу. Дякую
SamChen

Приємна відповідь. Я використовував "application / json" у моєму випадку
Gajendra K Chauhan

6

Я б запропонував використовувати ASIHTTPRequest

ASIHTTPRequest - це проста у використанні обгортка навколо API CFNetwork, яка полегшує деякі більш нудні аспекти спілкування з веб-серверами. Він написаний на Objective-C і працює як в Mac OS X, так і в додатках iPhone.

Він підходить для виконання базових запитів HTTP та взаємодії з послугами на основі REST (GET / POST / PUT / DELETE). Включений підклас ASIFormDataRequest дозволяє легко подавати дані та файли POST, використовуючи багаточастинні дані / дані форми


Зверніть увагу, що оригінальний автор припинив роботу з цим проектом. Дивіться наступне повідомлення щодо причин та альтернатив: http://allseeing-i.com/%5Brequest_release%5D ;

Особисто я великий шанувальник AFNetworking


3

Більшість з вас це вже знають, але я публікую це, на всякий випадок, деякі з вас все ще борються з JSON в iOS6 +.

У iOS6 та пізніших версіях ми маємо клас NSJSONSerialization, який є швидким і не залежить від включення "зовнішніх" бібліотек.

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

Це спосіб, яким iOS6 і пізніші версії можуть ефективно аналізувати JSON. Використання SBJson також є реалізацією до ARC і призводить до цих проблем, якщо ви працюєте в середовищі ARC.

Сподіваюся, це допоможе!


2

Ось чудова стаття з використанням Restkit

Тут пояснюється серіалізація вкладених даних у JSON та приєднання даних до запиту HTTP POST.


2

Оскільки моє редагування відповіді Майка G щодо модернізації коду було відхилено 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"];

0

Ось оновлений приклад, який використовує NSURLConnection + sendAsynchronousRequest: (10.7+, iOS 5+), Запит "Опублікувати" залишається таким же, як і з прийнятою відповіддю, і тут для ясності опущений:

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

Питання було про POST
Ахмад,

2
ні, перша частина запитання стосується асинхронності, і тут немає відповіді, яка відповідає на це. Вітаємо за голосів проти.
ауко

0

Ви можете спробувати цей код для рядка send 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];
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.