How to create functors?
Steven D'Aprano
steven at REMOVE.THIS.cybersource.com.au
Thu Aug 20 05:25:48 EDT 2009
On Thu, 20 Aug 2009 01:36:14 -0700, Paul Rubin wrote:
> Steven D'Aprano <steven at REMOVE.THIS.cybersource.com.au> writes:
>> As near as I can tell, a functor is just an object which is callable
>> like a function without actually being implemented as a function, e.g.:
>
> No it's not anything like that either, at least as I'm used to the term
> in programming or in mathematics. Maybe it's used other ways though.
According to Wikipedia, functor can be used as a synonym for "function
object":
http://en.wikipedia.org/wiki/Function_object
which is what I was thinking of. So it seems there are at least two
meanings for the word, neither of which seems to apply to this thread :)
> As I'm used to it, it's a feature of certain static type systems. The
> notion isn't that useful in Python
I find the Haskell page entirely opaque and unintelligible. Well, perhaps
not *entirely* opaque, but pretty close: it assumes a mathematical
sophistication that I don't think I even had when I was getting my maths
degree, let alone can remember two decades later. (Pity the poor VB
coders wondering what Haskell is good for...) The Wikipedia page is a
little better, but it's section on Examples is laughable -- the examples
are as unintelligible to this reader as the description before them.
But let me try an example to see if I've got it right:
class Int2StrFunctor:
def map1(self, n):
if type(n) is not int:
raise TypeError('argument must be an int')
return "-"*n
def map2(self, f):
if type(f) is not type(lambda: None):
raise TypeError('argument must be a function')
# assume f takes an int, and returns another int
def inner(n):
return self.map1(f(n))
return inner
The functor can take an int and return a string:
>>> F = Int2StrFunctor() # F is a functor
>>> F.map1(3)
'---'
It can also take a function (of int -> int) and return a new function
(int -> str):
>>> def myfunc(n):
... return n+2
...
>>> f = F.map2(myfunc)
>>> f(3)
'-----'
>>> f(4)
'------'
There's nothing special about the methods map1() and map2(), I could call
them anything I like, or even do this:
>>> def __call__(self, arg):
... if type(arg) is int:
... return self.map1(arg)
... else:
... return self.map2(arg)
...
>>> Int2StrFunctor.__call__ = __call__
>>>
>>> F(2)
'--'
>>> F(myfunc)(0)
'--'
There are some technical restrictions on functors, relating to the sorts
of functions and types (strictly "categories") they can accept,
presumably to make them mathematically well-behaved.
Have I got it correct?
--
Steven
More information about the Python-list
mailing list