Skip to main content

A decorator for adding memoization to functions and methods.

Project description

Memoiz

A decorator for adding memoization to functions and methods.

Introduction

Memoiz provides a function decorator that adds memoization to a function or method. It makes reasonable assumptions about how and if to cache the return value of a function or method based on the arguments passed to it.

Features

  • Use the Memoiz decorator on functions and methods.
  • A thread-safe cache.
  • Use any number of arguments or keyword arguments.
  • Support for parameter and return type hints.
  • Handles circular references in dictionaries, lists, sets, and tuples.
  • Support for common unhashable types (e.g., dict, list, set).
  • Selective cache entry removal.

Table of Contents

Installation

pip install memoiz

Usage

Apply Memoization to Class Methods

In this example you will use Memoiz to memoize the return value of the greeter.greet method and print the greeting.

from memoiz import Memoiz

# `cache` is a Python decorator.
cache = Memoiz()


class Greeter:

    def __init__(self):
        self.adv = "Very"
    
    # The `cache` decorator adds memoization capabilities to the `greet` method.
    @cache
    def greet(self, adj: str) -> str:
        return f"Hello, {self.adv} {adj} World!"


greeter = Greeter()

print("1:", cache._cache)

greeting = greeter.greet("Happy")

print("2:", greeting)
1: {}
2: Hello, Very Happy World!

As a continuation of the example, you will selectively clear cached articles using the cache.clear method.

greeter = Greeter()

print("1:", cache._cache)

greeting = greeter.greet("Happy")

print("2:", greeting)

greeting = greeter.greet("Cautious")

print("3:", greeting)

# The cache has memoized the two method calls.
print("4:", cache._cache)

# Clear the call to `greeter.greet` with the "Happy" argument.
#                          ⮶ args
cache.clear(greeter.greet, "Happy")
#                   ⮴ method

print("5:", cache._cache)

# Clear the call to `greeter.greet` with the `Cautious` argument.
cache.clear(greeter.greet, "Cautious")

# The cache is empty.
print("6:", cache._cache)
1: {}
2: Hello, Very Happy World!
3: Hello, Very Cautious World!
4: {<bound method Greeter.greet of <__main__.Greeter object at 0x7f486842fbe0>>: {(('Happy',), ()): 'Hello, Very Happy World!', (('Cautious',), ()): 'Hello, Very Cautious World!'}}
5: {<bound method Greeter.greet of <__main__.Greeter object at 0x7f486842fbe0>>: {(('Cautious',), ()): 'Hello, Very Cautious World!'}}
6: {}

Apply Memoization to Functions

In this example you will use Memoiz to memoize the return value of the greet function and print the greeting.

from memoiz import Memoiz

cache = Memoiz()


@cache
def greet(adj: str) -> str:
    return f"Hello, {adj} World!"


print("1:", cache._cache)

greeting = greet("Happy")

print("2:", greeting)
1: {}
2: Hello, Happy World!

As a continuation of the example, you will selectively clear cached articles using the cache.clear method.

print("1:", cache._cache)

greeting = greet("Happy")

print("2:", greeting)

greeting = greet("Cautious")

print("3:", greeting)

print("4:", cache._cache)

#                  ⮶ args
cache.clear(greet, "Happy")
#           ⮴ function

# The cached call using the "Happy" argument is deleted; however, the call using the "Cautious" is still present.
print("5:", cache._cache)

#                  ⮶ args
cache.clear(greet, "Cautious")
#           ⮴ function

# The cache is now empty.
print("6:", cache._cache)
1: {}
2: Hello, Happy World!
3: Hello, Cautious World!
4: {<function greet at 0x7f486842bd00>: {(('Happy',), ()): 'Hello, Happy World!', (('Cautious',), ()): 'Hello, Cautious World!'}}
5: {<function greet at 0x7f486842bd00>: {(('Cautious',), ()): 'Hello, Cautious World!'}}
6: {}

Memoization Strategy

Memoiz will attempt to recursively transform a callable's arguments into a hashable key. The key is used in order to index and look up the callable's return value. The strategy that Memoiz employs for key generation depends on the type of the argument(s) passed to the callable. The Type Transformations of Common Types table provides examples of how Memoiz transforms arguments of common types.

Type Transformations of Common Types

Type Example Hashable Representation
dict {'b':42, 'c': 57, 'a': 23} (('a', 23), ('b', 42), ('c', 57))
list [23, 42, 57] (23, 42, 57)
tuple (23, 42, 57) (23, 42, 57)
set {..., 23, "42", 57} (23, '42', 57, Ellipsis)
hashable types ... (Ellipsis,)

Dictionaries

By default a dictionary is sorted by the string representation of its keys prior to indexing the callable's return value.

Sets

By default a set is sorted by the string representation of its values prior to indexing the callable's return value.

API

The Memoiz Class

memoiz.Memoiz(iterables, mappables, sortables, deep_copy)

  • iterables Tuple[type, ...] An optional tuple of types that are assumed to be iterables. Default (list, tuple, set)
  • mappables Tuple[type, ...] An optional tuple of types that are assumed to be mappings. Default (dict, OrderedDict)
  • sortables Tuple[type, ...] An optional tuple of types that are sorted by the string representation of their keys or values prior to indexing the return value. Default (dict, set)
  • deep_copy bool Optionally return the cached return value using Python's copy.deepcopy. This can help prevent mutations of the cached return value. Default: True.

memoiz.__call__(callable)

  • callable typing.Callable The function or method for which you want to add memoization.

A Memoiz instance is a callable. This is the @cache decorator (see Usage above) that is used in order to add memoization to a callable.

memoiz.clear(callable, *args, **kwargs)

  • callable typing.Callable The callable.
  • args Any The arguments passed to the callable.
  • kwargs Any The keyword arguments passed to the callable.

Clears the cache for the specified callable and arguments. See Usage for for how to clear the cache.

memoiz.clear_all()

Resets the cache making items in the old cache potentially eligible for garbage collection.

memoiz.clear_callable(callable)

Clears the cache and all of the entries for the specified callable.

Versioning

The Memoiz package strictly adheres to semantic versioning. Breaking changes to the API will result in a turn of the major. Minor and patch changes will always be backward compatible.

Excerpted from Semantic Versioning 2.0.0:

Given a version number MAJOR.MINOR.PATCH, increment the:

  1. MAJOR version when you make incompatible API changes
  2. MINOR version when you add functionality in a backward compatible manner
  3. PATCH version when you make backward compatible bug fixes

Additional labels for pre-release and build metadata are available as extensions to the MAJOR.MINOR.PATCH format.

Test

Clone the repository.

git clone https://github.com/faranalytics/memoiz.git

Change directory into the root of the repository.

cd memoiz

Run the tests.

python tests/test.py -v

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

memoiz-4.0.2.tar.gz (8.2 kB view details)

Uploaded Source

Built Distribution

memoiz-4.0.2-py3-none-any.whl (6.7 kB view details)

Uploaded Python 3

File details

Details for the file memoiz-4.0.2.tar.gz.

File metadata

  • Download URL: memoiz-4.0.2.tar.gz
  • Upload date:
  • Size: 8.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: python-httpx/0.27.2

File hashes

Hashes for memoiz-4.0.2.tar.gz
Algorithm Hash digest
SHA256 65c224d533f904ef1c5cd0c6917165dc7c82470a9f08c7f6e6f81cae7ccf0687
MD5 206f7da7bfed278d3f8926140e84ed94
BLAKE2b-256 30bd197b3ccf81457b12b84a399f30a476bea8dc5d74ca24929950a7df8ab220

See more details on using hashes here.

File details

Details for the file memoiz-4.0.2-py3-none-any.whl.

File metadata

  • Download URL: memoiz-4.0.2-py3-none-any.whl
  • Upload date:
  • Size: 6.7 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: python-httpx/0.27.2

File hashes

Hashes for memoiz-4.0.2-py3-none-any.whl
Algorithm Hash digest
SHA256 da99961d076a723ac6b342dc3a6e19c2505b15f438501a46b7113c33f7ec855f
MD5 2efc475bf4f9d9bac204dd6814e0797e
BLAKE2b-256 b4c07a865fafb56c0b49b9c43f4ed55941c02ddca4f5dc18bb1282b43f761395

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page