Как отправить данные json в запрос Http с помощью NSURLRequest
Я новичок в objective-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];
}
Мой первый вопрос: будет ли этот подход расширяться? Или это не async (это означает, что я блокирую поток пользовательского интерфейса, пока приложение ждет ответа)
Мой второй вопрос: как я могу изменить часть запроса этого, чтобы сделать POST вместо GET? Это просто изменить HttpMethod так?
[request setHTTPMethod:@"POST"];
И, наконец, как добавить набор данных json на этот пост в виде простой строки (например)
{
"magic":{
"real":true
},
"options":{
"happy":true,
"joy":true,
"joy2":true
},
"key":"123"
}
Заранее благодарю
Ответы
Ответ 1
Вот что я делаю (обратите внимание, что 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% ясно и займет некоторое повторное чтение, но все должно быть здесь, чтобы вы начали. И из того, что я могу сказать, это асинхронно. Мой пользовательский интерфейс не заблокирован во время этих вызовов. Надеюсь, что это поможет.
Ответ 2
Я бы предложил использовать ASIHTTPRequest
ASIHTTPRequest - простой в использовании обертка вокруг API CFNetwork, которая делает некоторые из более утомительных аспектов связи с веб-серверами Полегче. Он написан в Objective-Cи работает как в Mac OS X, так и в iPhone приложения.
Подходит для выполнения базового HTTP-протокола запросов и взаимодействия с Службы на основе REST (GET/POST/PUT /УДАЛИТЬ). Включенный Подкласс ASIFormDataRequest делает это легко отправлять данные и файлы POST используя multipart/form-data.
Обратите внимание, что оригинальный автор прекратил этот проект. См. Последующую публикацию по причинам и альтернативам: http://allseeing-i.com/%5Brequest_release%5D;
Лично я большой поклонник AFNetworking
Ответ 3
Я боролся с этим некоторое время. Запуск 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
}
}];
Ответ 4
Большинство из вас уже это знают, но я публикую это, просто выставляю, некоторые из вас все еще борется с JSON в iOS6 +.
В iOS6 и более поздних версиях мы имеем класс NSJSONSerialization, который быстро и не имеет зависимости от включения "внешних" библиотек.
NSDictionary *result = [NSJSONSerialization JSONObjectWithData:[resultStr dataUsingEncoding:NSUTF8StringEncoding] options:0 error:nil];
Таким образом, iOS6 и более поздние версии теперь могут эффективно анализировать JSON. Использование SBJson также является пре-ARC-реализацией и приносит с собой эти проблемы, если вы работаете в среде ARC.
Надеюсь, это поможет!
Ответ 5
Вот отличная статья, используя Restkit
В нем объясняется сериализация вложенных данных в JSON и привязка данных к HTTP POST-запросу.
Ответ 6
Так как мое редактирование для Майка 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"];
Ответ 7
Здесь обновленный пример, в котором используется 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);
}
}
}];
Ответ 8
Вы можете попробовать этот код для отправки строки 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¶m=%@",jsonString];