Memoize in Swift

An explanation of a memoization implementation in swift

2014-10-13

Metadata
Memoize in Swift
An explanation of a memoization implementation in swift
2014-10-13
./swift.jpg
/2014/10/13/memoize-in-swift.html/2014/10/13/memoize-in-swift
swiftiosapple

I've been learning swift lately in my IOS class and was introduced to closures. After working on the [memoize]blog/memoize) code in javascript a few days ago I realized that I could implement a similar solution in swift. There are a few caveats though because of differences in the language and flexibility.

So here is the memoize function in swift:

func memoize(function:([Any]) -> Any?) -> (Any...) -> Any? {
  var cache = [String: Any]()

  return {(parameters:Any...) -> Any? in
    var key:String = ""
    if parameters.count > 0 {
      key = parameters.description
    }

    if (cache[key] != nil) {
      return cache[key]
    } else {
      var value = function(parameters)
      return value
    }
  }
}

So here is an explanation of the code. If you are familiar with my javascript solution, this will look very similar but there are a few differences so I will focus on those differences.

First of all the function signature is much different because of the pseudo-dynamic types in swift. Although variables can be typed at compile-time, it will eventually get typed, and once set, it can't change.

So let’s break down the signature.

func memoize(function:([Any]) -> Any?) -> (Any...) -> Any?

So the memoize function has a single parameter, a closure that takes an array of Any and returns an Any optional. The reason why it takes an array of Any will be more clear as I go further into the function.

The memoize function will return a closure of type Any variadic that returns an Any optional. The reason why the returned function takes an Any variadic is so that the function can be called with as many parameters as necessary.

For those unfamiliar, variadic parameters allow a function to take an unlimited number of parameters of the given type. So a function with a variadic parameter could be called foo(1) or foo(1, 2, 3). All of the parameters will be in an array of the parameter name. For more info on variadic parameters checkout Apple’s documentation.

The rest of the memoize function is pretty much the same as the javascript implementation except for the calling of the original function. I couldn't find a way to call a function in swift like you can with the javascript apply function so we just pass in the array of parameters captured in the variadic parameter. This is why the closure that the memoize function takes must have an input parameter of type [Any] because all of the parameters will be passed into it as an array. This is less than ideal but is the only solution that I could find.

So that's the just of the memoize function. In order to use the function, the closure you write must be a specific type.

Here is an example of a possible function that could be memoized.

var sum = {(parameters:[Any]) -> Any? in
    var result:Int = 0
    for num in parameters {
          result += num as Int
    }
    return result
}

Then we would memoize it by passing it to memoize. But because the closure that is returned from memoize is of a different type than the closure we are passing into memoize, we must assign the result to a new variable.

var memSum = memoize(sum)

So now if we run memSum:

memSum(1, 2) //returns 3 from the original function
memSum(1, 2) //returns 3 from the cache

So there you have it, a memoize function in swift. Although it may not be ideal to write your closures with the required type, it could prove beneficial at some point.