DeferredKit is an asynchronous library for cocoa built around the idea of a Deferred Object - that is, "an object created to encapsulate a sequence of callbacks in response to an object that may not yet be available." Besides the core class, DKDeferred, much other functionality is included in this project, including an asynchronous URL loading API, an asynchronous disk cache, and a JSON-RPC implementation.
DeferredKit is modeled after the deferred class by TwistedMatrix and inspired by MochiKit's implementation of Deferred. DKCallback - the function object is mostly taken from a pre-blocks version of FunctionalKit.
The DKDeferred implementation is not dependent upon threads or any other form of concurrency for it's operation (however, you may create threaded Deferred's) and operates in the same environment as the rest of your Objective-C program.
NOTE: DeferredKit bundles json-framework, and will need to be removed from your project before adding DeferredKit using the following method. Otherwise, embedding the code works just as well.
More:
- Copy the entire source tree into your projects directory.
- Add DeferredKit to your project.
- Copy
"{PROJECT_ROOT}/DeferredKit/CocoaDeferred/CocoaDeferred.xcodeproj"
- In the window presented by Xcode, uncheck "Copy items...". Reference type should be "Relative to Project"
- Uncheck any targets Xcode might automatically assume.
- Copy
- Add DeferredKit to your header search paths.
- Under your target's build settings, search for find "Header Search Paths" and add
"DeferredKit/CocoaDeferred/Source"
- Under your target's build settings, search for find "Header Search Paths" and add
- Add DeferredKit to your Target
- Under your target's general settings, under Direct Dependancies click the "+" button and choose "DeferredKit"
- Expand your
"CocoaDeferred.xcodeproj"
and drag"libDeferredKit.a"
to your target's "Link Binary with Library"
All methods in DeferredKit return Deferred objects. This is the same basic interface used to access all functionality provided by DeferredKit.
id cbGotResource(id results) {
[[Resource resourceWithData:results] save];
return nil;
}
id cbGetResourceFailed(id error) {
// alert user resource is unavailable.
return nil;
}
DKDeferred *d = [DKDeferred loadURL:@"http://addr.net/resource/"];
[d addCallback:callbackP(cbGotResource)];
[d addCallback:callbackP(cbGetResourceFailed)];
You can generate Deferred objects which encapsulate the execution of a method or function in a thread. The Deferred automatically returns the result to the correct thread.
id cbDoneProcessing(id results) {
if (content) {
[content release];
content = nil;
}
content = [results retain];
[tableView reloadData];
return nil;
}
DKDefered *d =[DKDeferred deferInThread:
callbackTS((id)[Resource class], updateAllResources:)];
[d addCallback:cbDoneProcessing];
These two Deferred objects may return almost immediately if loaded from the cache.
- (IBAction)loadResource:(id)sender {
DKDeferred *html = [DKDeferred loadURL:@"http://url1.com/resource" cached:YES];
DKDeferred *header = [DKDeferred loadImage:@"http://url1.com/resource-img.png" cached:YES];
DKDeferred *d = [DKDeferred gatherResults:array_(html, header)];
[d addCalback:callbackTS(self, cbDoneLoading:)];
}
- (id)cbDoneLoading:(id)results {
[self showHTML:[results objectAtIndex:0]];
[self showHeaderImage:[results objectAtIndex:1]];
return nil;
}
DeferredKit provides a JSON-RPC implementation using DKDeferred.
id myservice = [DKDeferred jsonService:@"" name:@"myservice"]
DKDeferred *d = [myservice someMethod:array(arg1, arg2)]
[d addCallbacks:callbackTS(self, cbGotResults:) :callbackTS(cbGetResultsFailed:)];
Each callback added to a DKDeferred results in a chain of callbacks - the last callback added will be called with the result returned by the previous callback.
- (IBAction)fetchResources:(id)sender {
id _parseResults:(id results) {
// _parseResults can return an NSError at which point the deferred
// will begin it's error callback chain
return [Resource arrayWithJSONResponse:results];
}
DKDeferred *d = [DKDeferred loadJSONDoc:@"http://whereitsat.net/resource/"]
[d addCallback:callbackP(_parseResults)];
[d addCallback:callbackTS(self, _presentResources:)];
[d addErrback:callbackTS(self, _getResourcesFailed:)];
}
- (id)_presentResources:(id)results {
if (resources) {
[resources release];
resources = nil;
}
resources = [results retain];
[tableView reloadData];
}
Since the disk cache utilizes a deferred object interface, access to cached results can implement caching in only a few lines.
- (IBAction)fetchSomeStuff:(id)sender {
id _gotKey(id results) {
if (results == [NSNull null]) { // cache miss
return [DKDeferred deferInThread:[Resource getResources] withObject:nil];
} else { // cache hit
return results;
}
}
DKDeferred *d = [[DKDeferredCache sharedCache] valueForKey:@"someKey"];
[d addCallback:callbackP(_gotKey)];
[d addCallback:callbackTS(self, cbGotResults:)];
}
- (id)cbGotResults:(id)results {
if (isDeferred(results)) // in the event of a cache miss
return [results addCallback:callbackTS(self, cbGotResults:)];
if (resources) {
[resources release];
resources = nil;
}
resources = [results retain];
[tableView reloadData];
}
Returns an empty
DKDeferred