Проект на google code
Запись в блоге с примерами использования
Библиотека динамической перегрузки функций и методов основанной на типах аргументов для языка pythonВозможно, кому то будет интересно поэкспериментировать с библиотекой:)
Библиотека overloading определяет средства позволяющие писать перегруженные функции и методы в вашем коде. Весь функционал реализован в единственном модуле overloading.py.
Так-же стоит добавить что библиотека была вдохновлена PEP 3124, в ней нет и половины возможностей и расширяемости первоисточника, но я писал её скорее для фана (однако в дальнейшем я использовал её в нескольких своих проектах). В моём представлении она должна выполнять свою основную задачу, добавлять в язык перегрузку функций, основанную на типах аргументов, максимально близкую к реализации оной из C++. Т.е. от идеи включения в библиотеку реализации интерфейсов и прочего функционала, я отказался, сосредоточившись на перегрузке. Так как писал я библиотеку на чистом python'е и поиск функций происходит в runtime, то старался максимально сократить время поиска среди кандидатов, подходящей функции (Phillip J. Eby, автор PEP`а и библиотеки PEAK для оптимизации использовал Bytecode Assembler, а это сильная магия:))
Для определения перегруженной функции/метода существуют декораторы overload, и when, overload служит для определения перегруженной функции, сигнатура задается в параметрах декоратора, пример:При разрешении перегрузки используется полиморфизм:@overload(float)
def func(x):
...
@overload(int)
def func(x):
...
func(1) # вызов функции с сигнатурой func(x: int)
func(1.0) # вызов функции с сигнатурой func(x: float)Декоратор overload в комбинации с mix-in классом Overloadable позволяет определять перегруженные методы, семантика аналогична использованию overload для функций (за исключением того что в сигнатуру метода не входит первый параметр self), пример:class MyStr(str): pass
@overload(str)
def func(x):
...
func("string") # вызов функции с сигнатурой func(x: str)
func(MyStr()) # вызов функции с сигнатурой func(x: str)Декоратор when позволяет определить перегруженную функцию/метод имя которой не совпадает с именем других перегруженных методов, пример:class T(Overloadable):
@overload(bool)
def meth(self, x):
...
@overload(int, str)
def meth(self, x, y):
...
i=T()
i.meth(1, 'hello') # вызов метода с сигнатурой meth(x: int, y: str)
i.meth(True) # вызов метода с сигнатурой meth(x: bool)
i.meth("Opps!") # исключение CannotResolve, метод с сигнатурой
# meth(x: str) не определенПервым параметром декоратора должна быть перегруженная функция/метод, определенная ранее в данном пространстве имен, второй параметр кортеж из типов аргументов функции.@overload(bool)
def func(x):
...
@overload(int)
def func(x):
...
@when(func, (list,))
def func_list(x):
...
func(1) # вызов функции с сигнатурой func(x: int)
func(True) # вызов функции с сигнатурой func(x: bool)
func([1,2,3]) # вызов функции определенной с помощью декоратора when
func_list([1]) # явный вызов функции
Определение обобщенной функции:Обобщенная функция должна быть определена раньше её специализированных вариантов и в том же пространстве имен# Обобщенная функция
def func(x):
...
# Специализированные функции
@overload(int)
def func(x):
...
@overload(float)
def func(x):
...
func(1) # вызов функции с сигнатурой func(x: int)
func(1.0) # вызов функции с сигнатурой func(x: float)
func([1,2,3]) # вызов обобщенной функции
Ограничения:
- Нет поддержки аргументов со значениями по умолчанию
- Нет поддержки переменного числа аргументов (*args)
- Нет поддержки именованных аргументов (**kwargs)
- Нет приведения типов, например int не будет приведен к float
- Декораторы overload и when несовместимы с любыми другими декораторами включая classmethod и staticmethod