Rest no iOS 5

Veja neste artigo como conectar-se a servidores web externos e a como tratar as informações recebidas

Este artigo trata sobre requisições http a servidores web externos. Essa técnica é útil para quem quer conectar-se com API’s externas, como a do Facebook ou a do Twitter. Vou abordar os seguintes assuntos: JSON, Categories, NSURL, NSData.

Programadores de iOS 4 devem estar acostumados com uma biblioteca JSON externa para desenvolvimento. Felizmente a Apple adicionou uma biblioteca nativa de JSON ao iOS 5. Mas vamos lá, mãos à obra.

Conexão com um servidor externo

Em Objective-C é muito fácil se conectar a um servidor externo via http, basta seguir os passos abaixo:

1 - Criar um objeto NSURL

NSURL é uma classe que guarda URL’s (Uniform Resource Locator) tanto para servidores externos quanto para diretórios no filesystem do iPhone. Nesse caso, vamos usar o método urlWithString, que gera um objeto NSURL a partir de uma string com o link.

NSURL* url = [NSURL urlWithString: @”http://seu-web-site.com/api/algo”]; Se você quiser ver como ficou a requisição, pode dar uma série de NSLogs: NSLog(@"Scheme: %@", [url scheme]); NSLog(@"Host: %@", [url host]); NSLog(@"Port: %@", [url port]); NSLog(@"Path: %@", [url path]); NSLog(@"Relative path: %@", [url relativePath]); NSLog(@"Path components as array: %@", [url pathComponents]); NSLog(@"Parameter string: %@", [url parameterString]); NSLog(@"Query: %@", [url query]); NSLog(@"Fragment: %@", [url fragment]);

2 - Criar um NSData

NSData é um objeto do Foundation framework e representa uma série de bytes. O NSData não faz ideia se aquela sequencia de bytes forma um número, uma string ou qualquer outra coisa.

NSData possui um método de classe chamado: dataWithContentsOfUrl que faz a requisição http, pega e guarda a resposta em um objeto. Esse método leva como argumento a URL que criamos anteriormente, que é aonde ele vai conectar-se.

NSData *data = [NSData dataWithContentOfUrl:url]; Pronto, dentro desse NSData já temos uma resposta em JSON, agora só falta transformar o conteúdo da resposta em algo legível para nós.

A biblioteca JSON no iOS 5

O nome da classe que faz a leitura do JSON é NSJSONSerialization. Vamos ver como essa classe funciona. Para fazer isso vamos usar um método chamado JSONObjectWithData:options:error: que transforma um objeto NSData em um NSDictionary ou um NSArray. Nesse tutorial vamos usar um NSDictionary.

NSDictionary

NSDictionary é uma estrutura de dados que armazena valores a partir de uma KEY, podemos comparar com o std::map de C++, ou talvez até com um $array[“key”] do PHP. Para acessar seus dados, usamos o método objectForKey.

1 - Transformação de NSData para NSDictionary

NSError* error; NSDictionary* json = [NSJSONSerialization JSONObjectWithData: data options: kNilOptions error: &error]; Como podem ver no código acima, estamos transformando um NSData em um NSDictionary. O método JSONObjectWithData leva 3 argumentos: o objeto NSData que ele vai transformar, options que são as opções de transformação e uma variável para armazenar algum erro, caso aconteça. A constante kNilOptions é uma constante para SEM OPÇÕES.

2 - Leitura do Dictionary

Cada chave no NSDictionary representa um objeto do JSON, e se esse objeto contém mais objetos dentro dele, um novo dictionary é criado, veja o exemplo abaixo:

{ "webcam": { // objeto “webcam” com 3 objetos dentro dele "webcamid": "1234", "title": "Foo", "owner": "Bar" } } NSDictionary *dict = [json objectForKey:@”webcam”]; // O objeto webcam é um dictionary NSString* title= [dict objectForKey: @”title”]; // objeto title dentro do objeto webcam NSLog(title); // Vai logar Foo Como podem perceber, dentro do objeto “webcam” há outros objetos, portanto ele retorna um dictionary. A partir daí, objetos normais são strings ou números.

Criando uma categoria para facilitar o desenvolvimento

Vamos fazer uma CATEGORIA na classe NSDictionary, uma categoria é um artifício em que você pode adicionar funções a uma classe sem herdá-la.

1 - Criar o arquivo da categoria

No XCode vá a File -> New -> New File e no menu Cocoa Touch escolha: Objective-C category.



Agora haverá dois formulários assim:

Category:
Category on:
Isso é o nome da categoria e a classe em que quer colocá-la, no nosso caso chamaremos de JSONCategory e a colocaremos na classe NSDictionary.



Será criado um arquivo chamado NomeDaClasse + NomeDaCategoria e você vai ter algo parecido com isso:

NSDictionary+JSONCategory.h
@interface NSDictionary (JSONCategory) @end NSDictionary+JSONCategory.m
@implementation NSDictionary (JSONCategory) @end Vamos adicionar as seguintes funções ao NSDictionary+JSONCategory.h:
+(NSDictionary*)dictionaryWithContentsOfJSONURLString: (NSString*)urlAddress; -(NSData*) toJSON; O método dictionaryWithContentsOfJSONURLString recebe um argumento com o endereço do URL da API em que você quer conectar-se.

O método toJSON transforma o dicionário em NSData com um JSON.

Agora vamos implementar essas funções:
+(NSDictionary*)dictionaryWithContentsOfJSONURLString: (NSString*)urlAddress { NSData* data = [NSData dataWithContentsOfURL: [NSURL URLWithString: urlAddress] ]; // Pega a data e a resposta __autoreleasing NSError* error = nil; //Variavel que guardara o erro id result = [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:&error]; // Faz le o resultado do JSON if (error != nil) return nil; return result; // Retorna o dictionary } -(NSData*)toJSON { NSError* error = nil; id result = [NSJSONSerialization dataWithJSONObject:self options:kNilOptions error:&error]; // Transforma um dictionary em um NSData com JSON if (error != nil) return nil; return result; } A única coisa nova apresentada nessas implementações é o método:
dataWithJSONObject que transforma o dictionary em um objeto NSData contendo os bytes do json criado a partir do dictionary.

Exemplo de como usar:

NSDictionary dict = [NSDictionary dictionaryWithObjectsAndKeys: @”hey”,@”hey2”,nil]; NSData* data = [dict toJSON]; // Pronto, você tem uma NSData com JSON pronta para enviar a algum servidor. Bem, o útlimo método que eu queria comentar é o: isValidJSONObject: que verifica se algum objeto pode ser transformado em JSON, por exemplo: BOOL ableToBeJson = [JSONSerialization isValidJsonObject: object]; // retorna YES ou NO dependendo se aquele objeto está apto //para virar um JSON ou não

Conclusão


Este artigo foi criado por Henrique Dubugras.

Contato: dubugras@ingresse.com

Artigos relacionados