Как размножать функции в python?
def sub3(n):
return n - 3
def square(n):
return n * n
Он мертв легко компилирует функции в python:
>>> my_list
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
>>> [square(sub3(n)) for n in my_list]
[9, 4, 1, 0, 1, 4, 9, 16, 25, 36]
К сожалению, когда вы хотите использовать композицию в качестве ключа, она выглядит как lame:
>>> sorted(my_list, key=lambda n: square(sub3(n)))
[3, 2, 4, 1, 5, 0, 6, 7, 8, 9]
Это должно быть просто sorted(my_list, key=square*sub3)
, потому что heck, function __mul__
не используется ни для чего другого:
>>> square * sub3
TypeError: unsupported operand type(s) for *: 'function' and 'function'
Ну давайте просто определим его!
>>> type(sub3).__mul__ = 'something'
TypeError: can't set attributes of built-in/extension type 'function'
D'ах!
>>> class CoolerFunction(types.FunctionType):
... pass
...
TypeError: Error when calling the metaclass bases
type 'function' is not an acceptable base type
D'о!
class Hack(object):
def __init__(self, function):
self.function = function
def __call__(self, *args, **kwargs):
return self.function(*args, **kwargs)
def __mul__(self, other):
def hack(*args, **kwargs):
return self.function(other(*args, **kwargs))
return Hack(hack)
Эй, теперь мы куда-то...
>>> square = Hack(square)
>>> sub3 = Hack(sub3)
>>> [square(sub3(n)) for n in my_list]
[9, 4, 1, 0, 1, 4, 9, 16, 25, 36]
>>> [(square*sub3)(n) for n in my_list]
[9, 4, 1, 0, 1, 4, 9, 16, 25, 36]
>>> sorted(my_list, key=square*sub3)
[3, 2, 4, 1, 5, 0, 6, 7, 8, 9]
Но я не хочу класс Hack
вызываемый класс! Правила охвата совершенно разные, как я не совсем понимаю, и это даже уродливее, чем "ламеда". Я хочу, чтобы monkeypatch функции. Как я могу это сделать?
Ответы
Ответ 1
Вы можете использовать свой класс хака как декоратор в значительной степени, как он написал, хотя вы, вероятно, захотите выбрать более подходящее имя для класса.
Вот так:
class Composable(object):
def __init__(self, function):
self.function = function
def __call__(self, *args, **kwargs):
return self.function(*args, **kwargs)
def __mul__(self, other):
@Composable
def composed(*args, **kwargs):
return self.function(other(*args, **kwargs))
return composed
def __rmul__(self, other):
@Composable
def composed(*args, **kwargs):
return other(self.function(*args, **kwargs))
return composed
Затем вы можете украсить свои функции следующим образом:
@Composable
def sub3(n):
return n - 3
@Composable
def square(n):
return n * n
И составьте их так:
(square * sub3)(n)
В основном это то же самое, что вы сделали, используя свой класс взлома, но используя его как декоратор.
Ответ 2
Python не поддерживает (и, скорее всего, никогда) поддержку композиции функций на синтаксическом уровне или в качестве стандартной библиотечной функции. Существуют различные сторонние модули (такие как functional), которые обеспечивают функцию более высокого порядка, которая реализует композицию функций.
Ответ 3
Возможно, что-то вроде этого:
class Composition(object):
def __init__(self, *args):
self.functions = args
def __call__(self, arg):
result = arg
for f in reversed(self.functions):
result = f(result)
return result
И затем:
sorted(my_list, key=Composition(square, sub3))
Ответ 4
Вы можете создавать функции, используя библиотеку SSPipe:
from sspipe import p, px
sub3 = px - 3
square = px * px
composed = sub3 | square
print(5 | composed)