Multiple argument dispatching.
Project description
Multimethod provides a decorator for adding multiple argument dispatching to functions. The decorator creates a multimethod object as needed, and registers the function with its annotations.
There are several multiple dispatch libraries on PyPI. This one aims for simplicity and speed. With caching of argument types, it should be the fastest pure Python implementation possible.
Usage
multimethod
from multimethod import multimethod
@multimethod
def func(x: int, y: float):
...
func is now a multimethod which will delegate to the above function, when called with arguments of the specified types. Subsequent usage will register new types and functions to the existing multimethod of the same name.
@multimethod
def func(x: float, y: int):
...
Alternatively, functions can be explicitly registered in the same style as functools.singledispatch. This syntax is also compatible with mypy, which by default checks that each name is defined once.
@func.register
def _(x: bool, y: bool):
...
@func.register(object, bool)
@func.register(bool, object)
def _(x, y): # stackable without annotations
...
Multimethods are implemented as mappings from signatures to functions, and can be introspected as such.
method[type, ...] # get registered function
method[type, ...] = func # register function by explicit types
Multimethods support any types that satisfy the issubclass relation, including abstract base classes in collections.abc and typing. Subscripted generics are provisionally supported:
Union[...]Mapping[...]- the first key-value pair is checkedTuple[...]- all args are checkedIterable[...]- the first arg is checked
Naturally checking subscripts is slower, but the implementation is optimized, cached, and bypassed if no subscripts are in use in the multimethod.
Dispatch resolution details:
- If an exact match isn't registered, the next closest method is called (and cached).
- If the
issubclassrelation is ambiguous, mro position is used as a tie-breaker. - If there are still ambiguous methods - or none - a custom
TypeErroris raised. - Default and keyword-only parameters may be annotated, but won't affect dispatching.
- A skipped annotation is equivalent to
: object, which implicitly supports methods by leavingselfblank. - If no types are specified, it will inherently match all arguments.
overload
Overloads dispatch on annotated predicates. Each predicate is checked in the reverse order of registration.
The implementation is separate from multimethod due to the different performance characteristics. Instead a simple isa predicate is provided for checking instance type.
from multimethod import isa, overload
@overload
def func(obj: isa(str)):
...
@overload
def func(obj: str.isalnum):
...
@overload
def func(obj: str.isdigit):
...
multimeta
Use metaclass=multimeta to create a class with a special namespace which converts callables to multimethods, and registers duplicate callables with the original.
from multimethod import multimeta
class Foo(metaclass=multimeta):
def bar(self, x: str):
...
def bar(self, x: int):
...
Equivalent to:
from multimethod import multimethod
class Foo:
@multimethod
def bar(self, x: str):
...
@bar.register
def bar(self, x: int):
...
Installation
% pip install multimethod
Tests
100% branch coverage.
% pytest [--cov]
Changes
1.5
- Postponed evaluation of nested annotations
- Variable-length tuples of homogeneous type
- Ignore default and keyword-only parameters
- Resolved ambiguous
Uniontypes - Fixed an issue with name collision when defining a multimethod
- Resolved dispatch errors when annotating parameters with meta-types such as
type
1.4
- Python >=3.6 required
- Expanded support for subscripted type hints
1.3
- Python 3 required
- Support for subscripted ABCs
1.2
- Support for typing generics
- Stricter dispatching consistent with singledispatch
1.1
- Fix for Python 2 typing backport
- Metaclass for automatic multimethods
1.0
- Missing annotations default to object
- Removed deprecated dispatch stacking
0.7
- Forward references allowed in type hints
- Register method
- Overloads with predicate dispatch
0.6
- Multimethods can be defined inside a class
0.5
- Optimized dispatching
- Support for
functools.singledispatchsyntax
0.4
- Dispatch on Python 3 annotations
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file multimethod-1.5.tar.gz.
File metadata
- Download URL: multimethod-1.5.tar.gz
- Upload date:
- Size: 11.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.3.0 pkginfo/1.7.0 requests/2.25.1 setuptools/52.0.0 requests-toolbelt/0.9.1 tqdm/4.56.0 CPython/3.8.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b9c6f85ecf187f14a3951fff319643e1fac3086d757dec64f2469e1fd136b65d
|
|
| MD5 |
cc6f8ee77975c9b575217851f4b98b8c
|
|
| BLAKE2b-256 |
8f5ab3daeadfaec5ff56aaa1d1d327ca2605c0a44f9d69ad5ceaa93abd5bfeec
|
File details
Details for the file multimethod-1.5-py3-none-any.whl.
File metadata
- Download URL: multimethod-1.5-py3-none-any.whl
- Upload date:
- Size: 7.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.3.0 pkginfo/1.7.0 requests/2.25.1 setuptools/52.0.0 requests-toolbelt/0.9.1 tqdm/4.56.0 CPython/3.8.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
889bcc8feb34181fbfa70f78e92f4c1927f0361e5decef0d2770b6ed7984c41c
|
|
| MD5 |
4bad80a1a23736016be6964926e28385
|
|
| BLAKE2b-256 |
134e899c33da18671c2cf47f0d43b232fc220d465e37e90d7a151c261779416b
|