iPhone JSON Library
This tutorial uses Stig Brautaset’s JSON library (version 2.2), which provides functionality for parsing and generating JSON. We won’t be using the JSON generating functionality in this tutorial.
The library provides two methods for parsing JSON: (1) a high-level category that extends NSString to include JSON parsing and (2) a more granular, slighly lower level object-based parser. We’ll start simple and use the former; we’ll switch to the latter near the end of the tutorial.
- Download the disk image
We’ll include the relevant bits into our project in a later step.
Creating The Project
Launch Xcode and create a new View-Based iPhone application called LuckyNumbers:
- Create a new project using File > New Project… from Xcode’s menu
- Select View-Based Application from the iPhone OS > Application section, click Choose…
- Name the project as LuckyNumbers and click Save
Adding JSON Support To The Project
In order to use the JSON functionality we’ll need to add it to the project:
- Expand the LuckyNumbers project branch in the Groups & Files panel of the project
- Using Finder, locate the JSON_2.2.dmg file you downloaded earlier and mount it by double clicking its icon. A new finder window with the DMG’s contents will open
- Drag and drop the JSON directory from the DMG onto the Classes folder under theLuckyNumbers project icon in the Groups & Files panel in Xcode
We’ll test that the JSON is set up properly by parsing a JSON dictionary string and using the resulting NSDictionary as a datasource for a message we’ll write to the console from our app’s
viewDidLoad
. This is throw-away code just to test that everything’s wired up properly.Use this code for LuckyNumbersViewController.m (located in the project’s Classes Xcode folder):
#import "LuckyNumbersViewController.h"
#import "JSON/JSON.h"
@implementation LuckyNumbersViewController
- (void)viewDidLoad {
[super viewDidLoad];
NSString *jsonString = [NSString stringWithString:@"{\"foo\": \"bar\"}"];
NSDictionary *dictionary = [jsonString JSONValue];
NSLog(@"Dictionary value for \"foo\" is \"%@\"", [dictionary objectForKey:@"foo"]);
}
- (void)dealloc {
[super dealloc];
}
@end
Build and run the project. If the JSON SDK is configured properly you should see an entry in the console that says, Dictionary value for “foo” is “bar”.
Setting Up The (Spartan) UI
Our finished app will need a UILabel to display the lucky numbers that we’ll eventually grab using HTTP and JSON.
Use this code for LuckyNumbersViewController.h (located in the project’s Classes Xcode folder):
#import <UIKit/UIKit.h>
@interface LuckyNumbersViewController : UIViewController {
IBOutlet UILabel *label;
}
@end
IBOutlet
is a macro that tells the compiler that it needs to wire up this variable to a corresponding UILabel
element added WYSIWYG in Interface Builder. In the next step, we’ll add that element and connect the two pieces:Edit the LuckyNumbersViewController.xib file in Interface Builder:
- Expand the Resources folder under the LuckyNumbers project branch in the Groups & Files panel.
- Double-click the LuckyNumbersViewController.xib file
Make sure that the Library, Inspector and View windows are open/visible. If they are not:
- Show the Library window using Tools > Library from the menu
- Show the Inspector window using Tools > Inspector from the menu
- Show the View by clicking the View icon on the LuckyNumbersViewController.xib window
Add the label:
- Locate the Label component in the Library window and drag it onto the view
- In the View window use the label’s handles to enlarge it to about half the size of the view
- In the Inspector window under View Attributes (the left-most tab), set the label’s number of lines to 0.
Setting the label’s number of lines to zero configures the label dynamically size the text to fit within its bounds.
Connect the Interface Builder label the code’s
label
. While still in Interface Builder:
- Control-click on File’s Owner icon in the LuckyNumbersViewController.xib window
- In the resulting pop-up menu, click-and-hold (i.e., don’t unclick) on the right-justified circle in the locationLabel row of the Outlets section
- Drag the mouse to the Label in the View. A blue line will connect the two.
Confirm that the two have been connected. The pop-up menu should look like the image on the right.
If everything looks right, save the changes and close Interface Builder.
Fetching JSON Over HTTP
We’ll use Cocoa’s NSURLConnection to issue an HTTP request and retrieve the JSON data.
Cocoa provides both synchronous and asynchronous options for making HTTP requests. Synchronous requests run from the application’s main runloop cause the app to halt while it waits for a response. Asynchronous requests use callbacks to avoid blocking and are straightforward to use. We’ll use asynchronous requests.
First thing we need to do is update our view controller’s interface to include an
NSMutableData
to hold the response data. We declare this in the interface (and not inside a method) because the response comes back serially in pieces that we stitch together rather than in a complete unit.Update LuckNumbersViewController.h. Changes are in bold:
#import <UIKit/UIKit.h>
@interface LuckyNumbersViewController : UIViewController {
IBOutlet UILabel *label;
NSMutableData *responseData;
}
To keep things simple, we’ll kick off the HTTP request from
viewDidLoad
.Replace the contents of LuckyNumbersViewController.m with:
#import "LuckyNumbersViewController.h"
#import "JSON/JSON.h"
@implementation LuckyNumbersViewController
- (void)viewDidLoad {
[super viewDidLoad];
responseData = [[NSMutableData data] retain];
NSURLRequest *request = [NSURLRequest requestWithURL:[NSURL URLWithString:@"http://www.unpossible.com/misc/lucky_numbers.json"]];
[[NSURLConnection alloc] initWithRequest:request delegate:self];
}
- (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 {
label.text = [NSString stringWithFormat:@"Connection failed: %@", [error description]];
}
- (void)connectionDidFinishLoading:(NSURLConnection *)connection {
[connection release];
}
- (void)dealloc {
[super dealloc];
}
@end
This mostly boilerplate code initializes the
responseData
variable to be ready to hold the data and kicks off the connection in viewDidload; it gathers the pieces as they come in indidReceiveData
; and the empty connectionDidFinishLoading
stands ready to do something with the results.Using The JSON Data
Next, we’ll flesh out the
connectionDidFinishLoading
method to make use of the JSON data retrieved in the last step.Update the
connectionDidFinishLoading
method in LuckyNumbersViewController.m. Use this code:
- (void)connectionDidFinishLoading:(NSURLConnection *)connection {
[connection release];
NSString *responseString = [[NSString alloc] initWithData:responseData encoding:NSUTF8StringEncoding];
[responseData release];
NSArray *luckyNumbers = [responseString JSONValue];
NSMutableString *text = [NSMutableString stringWithString:@"Lucky numbers:\n"];
for (int i = 0; i < [luckyNumbers count]; i++)
[text appendFormat:@"%@\n", [luckyNumbers objectAtIndex:i]];
label.text = text;
}
Our earlier throw-away test code created an NSDictionary. Here it creates an NSArray. The parser is very flexible and returns objects — including nested objects — that appropriately match JSON datatypes to Objective-C datatypes.
Better Error Handling
Thus far, we’ve been using the the convenient, high-level extensions to NSString method of parsing JSON. We’ve done so with good reason: it’s handy to simple send the
JSONValue
message to a string to accessed to the parsed JSON values.Unfortunately, using this method makes helpful error handling difficult. If the JSON parser fails for any reason it simply returns a nil value. However, if you watch your console log when this happens, you’ll see messages describing precisely what caused the parser to fail.
It’d be nice to be able to pass those error details along to the user. To do so, we’ll switch to the second, object-oriented method, that the JSON SDK supports.
Update the
connectionDidFinishLoading
method in LuckyNumbersViewController.m. Use this code:
- (void)connectionDidFinishLoading:(NSURLConnection *)connection {
[connection release];
NSString *responseString = [[NSString alloc] initWithData:responseData encoding:NSUTF8StringEncoding];
[responseData release];
NSError *error;
SBJSON *json = [[SBJSON new] autorelease];
NSArray *luckyNumbers = [json objectWithString:responseString error:&error];
[responseString release];
if (luckyNumbers == nil)
label.text = [NSString stringWithFormat:@"JSON parsing failed: %@", [error localizedDescription]];
else {
NSMutableString *text = [NSMutableString stringWithString:@"Lucky numbers:\n"];
for (int i = 0; i < [luckyNumbers count]; i++)
[text appendFormat:@"%@\n", [luckyNumbers objectAtIndex:i]];
label.text = text;
}
}
Using this method gives us a pointer to the error object of the underlying JSON parser that we can use for more useful error handling.