How do I write a generic memoize function?

后端 未结 15 1629
情话喂你
情话喂你 2020-12-09 05:25

I\'m writing a function to find triangle numbers and the natural way to write it is recursively:

function triangle (x)
   if x == 0 then return 0 end
   retu         


        
相关标签:
15条回答
  • 2020-12-09 06:21

    Recursion isn't necessary. The nth triangle number is n(n-1)/2, so...

    public int triangle(final int n){
       return n * (n - 1) / 2;
    }
    
    0 讨论(0)
  • 2020-12-09 06:22

    In the vein of posting memoization in different languages, i'd like to respond to @onebyone.livejournal.com with a non-language-changing C++ example.

    First, a memoizer for single arg functions:

    template <class Result, class Arg, class ResultStore = std::map<Arg, Result> >
    class memoizer1{
    public:
        template <class F>
        const Result& operator()(F f, const Arg& a){
            typename ResultStore::const_iterator it = memo_.find(a);
            if(it == memo_.end()) {
                it = memo_.insert(make_pair(a, f(a))).first;
            }
            return it->second;
        }
    private:
        ResultStore memo_;
    };
    

    Just create an instance of the memoizer, feed it your function and argument. Just make sure not to share the same memo between two different functions (but you can share it between different implementations of the same function).

    Next, a driver functon, and an implementation. only the driver function need be public int fib(int); // driver int fib_(int); // implementation

    Implemented:

    int fib_(int n){
        ++total_ops;
        if(n == 0 || n == 1) 
            return 1;
        else
            return fib(n-1) + fib(n-2);
    }
    

    And the driver, to memoize

    int fib(int n) {
        static memoizer1<int,int> memo;
        return memo(fib_, n);
    }
    

    Permalink showing output on codepad.org. Number of calls is measured to verify correctness. (insert unit test here...)

    This only memoizes one input functions. Generalizing for multiple args or varying arguments left as an exercise for the reader.

    0 讨论(0)
  • 2020-12-09 06:24

    In Scala (untested):

    def memoize[A, B](f: (A)=>B) = {
      var cache = Map[A, B]()
    
      { x: A =>
        if (cache contains x) cache(x) else {
          val back = f(x)
          cache += (x -> back)
    
          back
        }
      }
    }
    

    Note that this only works for functions of arity 1, but with currying you could make it work. The more subtle problem is that memoize(f) != memoize(f) for any function f. One very sneaky way to fix this would be something like the following:

    val correctMem = memoize(memoize _)
    

    I don't think that this will compile, but it does illustrate the idea.

    0 讨论(0)
提交回复
热议问题