Giter Club home page Giter Club logo

functionalkit's Introduction

FunctionalKit: It's Functional for Objective-C.

FunctionalKit is an attempt to use functional paradigms in Objective-C. It is a set of low level functional types & APIs. It contains types such as either, option, etc. that allow your to write correct, clean, tight, succinct and (where possible) typesafe code. It also provides more advanced concepts such as lifting functions into monads.

FunctionalKit is loosely modelled on Functional Java.

Setup/Installation

  1. Bring this project's structure into your project's structure, a git submodule is a good start. We place it into Source/External/functionalkit.
  2. Add functionalkit.xcodeproj to your project.
  3. For each of your targets, add FunctionalKit's functionalkit target as a direct dependency of the target (Target -> Command-I -> General -> Direct Dependencies).
  4. For each of your targets, ensure libfunctionalkit.a is included (linked against).
  5. In your project, add FunctionalKit's main directory to your Header Search Paths (HEADER_SEARCH_PATHS), e.g. $(SRCROOT)/Source/External/functionalkit/Source/Main.
  6. Import FK/FKPrelude.h where you want to use FunctionalKit, your prefix header is a good spot.

Usage

Examples

Function creation

Create a function from a selector.

id <FKFunction> doSomethingFunction = [FKFunction functionFromSelector:@selector(doSomething:) target:self];

Use a nested function (requires -fnested-functions):

// With -fnested-functions enabled.

  • (void)testDummy { NSString *f(NSString *a) { return [a substringToIndex:1]; } NSDictionary *d = [NSARRAY(@"one", @"two", @"three") groupByKey:functionP(f)]; }

Mapping

Map across the elements of an array of names and turn them into people.

NSArray *names = [NSArray arrayWithObject:@"Fred", @"Mary", @"Pete", nil];
NSArray *people = [names map:[FKFunction functionFromSelector:@selector(makePersonFromName:) target:self]];

// Given each chapter has an array of pages
NSArray *chapters = ...;
int pageCount = [[NSArray concat:[chapters map:functionS(pages)]] count];

Nil values

Handle a possibly nil value safely.

NSDictionary *dict = ...;
FKOption *couldBeNil = [KFOption fromNil:[dict objectForKey:@"SomeKey"]];

Handling failures

Construct an either representing failure.

FKEither *failed = [FKEither errorWithReason:@"Credentials have not been saved"];

Perform an operation that may fail, apply a selector to the result if successful, otherwise on failure propagate the error.

MVEither *maybeResponse = [HttpApi makeRequestToUrl:@"http://www.foo.com/json-api"];
MVEither *maybeParsedJson = [maybeResponse.right mapWithSelector:@selector(JSONValue)];

Perform an operation that returns nil & an error on failue. Return the error on the left on failure or the returned object on the right.

NSError *error;
SBJSON *parser = [[SBJSON new] autorelease];
id parsedObject = [parser objectWithString:self error:&error];
FKEither maybeParsedObject = [[FKOption fromNil:parsedObject] toEither:error];

Validate parsed values, turn them into a domain model class on success

Note. This is a bit messy, could be cleaner.

FKOption *maybeTitle = [FKOption fromNil:[dictionary objectForKey:@"title"] ofType:[NSString class]];
FKOption *maybeOwnerName = [FKOption fromNil:[dictionary objectForKey:@"owner_name"] ofType:[NSString class]];
FKOption *maybeHeadlineImgId = [FKOption fromNil:[dictionary objectForKey:@"headline_img_id"] ofType:[NSString class]];
if ([NSARRAY(maybeTitle, maybeOwnerName, maybeHeadlineImgId) all:@selector(isSome)]) {
	return [FKOption some:[FlickrGallery galleryWithTitle:[maybeTitle some] ownerName:[maybeOwnerName some] sampleImgId:[maybeHeadlineImgId some]]];
} else {
	return [FKOption none];
}

Side effects

Comap a function with an effect, to have the function execute then perform a side effect using the function's result.

id <FKFunction> getPhotosF = [FKFunction functionFromSelector:@selector(photos)];
id <FKEffect> galleriesOp = [FKEffect comap:[self effectThatDoesSomethingWithPhotos] :getPhotosF];

Fold and intersperse

// Declare some distinct search terms.
NSArray *searchTerms = NSARRAY(@"foo", @"bar", @"baz");

// Intersperse "OR" between each term, and flatten into a single string.
NSString *term = [[searchTerms intersperse:@" OR "] foldLeft:@"" f:[NSString concatF]];
NSLog(term);  // @"foo OR bar OR baz"

Lifting

The following example lifts a function into the array monad, applying the function to each element of the array.

// Parse a photo's details out of a dictionary, return Some(FlickrPhoto) on success or None on failure.
- (FKOption *)parsePhotoForDictionary:(NSDictionary *)dictionary {
	FKOption *maybeTitle = [FKOption fromNil:[dictionary objectForKey:@"title"] ofType:[NSString class]];
	FKOption *maybeId = [FKOption fromNil:[dictionary objectForKey:@"photo_id"] ofType:[NSString class]];
	if ([NSARRAY(maybeTitle, maybeId) all:@selector(isSome)]) {
		return [FKOption some:[FlickrPhoto photoWithId:[maybeId some] title:[maybeTitle some]]];
	} else {
		return [FKOption none];
	}
}

// Retrieve the array of photos.
FKOption *maybePhotos = [FKOption fromNil:[dictionary objectForKey:@"photos"] ofType:[NSArray class]];

// Still within the option monad, lift the parse function (above) into the array monad and map across the option.
id <FKFunction> parsePhotoF = [FKFunction functionFromSelector:@selector(parsePhotoForDictionary:) target:self];
FKOption *maybeParsedPhotos = [maybePhotos map:[NSArray liftFunction:parsePhotoF]];

Other Work

Here's some other work we know of that does similar things to FunctionalKit.

functionalkit's People

Contributors

0xced avatar nkpart avatar tomjadams avatar

Stargazers

 avatar

Watchers

 avatar  avatar

Recommend Projects

  • React photo React

    A declarative, efficient, and flexible JavaScript library for building user interfaces.

  • Vue.js photo Vue.js

    ๐Ÿ–– Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.

  • Typescript photo Typescript

    TypeScript is a superset of JavaScript that compiles to clean JavaScript output.

  • TensorFlow photo TensorFlow

    An Open Source Machine Learning Framework for Everyone

  • Django photo Django

    The Web framework for perfectionists with deadlines.

  • D3 photo D3

    Bring data to life with SVG, Canvas and HTML. ๐Ÿ“Š๐Ÿ“ˆ๐ŸŽ‰

Recommend Topics

  • javascript

    JavaScript (JS) is a lightweight interpreted programming language with first-class functions.

  • web

    Some thing interesting about web. New door for the world.

  • server

    A server is a program made to process requests and deliver data to clients.

  • Machine learning

    Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.

  • Game

    Some thing interesting about game, make everyone happy.

Recommend Org

  • Facebook photo Facebook

    We are working to build community through open source technology. NB: members must have two-factor auth.

  • Microsoft photo Microsoft

    Open source projects and samples from Microsoft.

  • Google photo Google

    Google โค๏ธ Open Source for everyone.

  • D3 photo D3

    Data-Driven Documents codes.