According to http://docs.python.org/2/library/functions.html#super,
If the second argument is omitted, the super object returned is unbound.
To explain terms bound / unbound I will use functions, and — to be brief — I give up details and caveats.
There are 2 views:
Free function (unbound), for example the built-in function sum()
:
sum([1, 2, 3]) # 6
A function bound to an object (other name: object method) means that a user have to bind it to a particular object with the dot (.
) notation.
For example the use of built-in .upper()
method:
"Alice".upper() # ALICE
"Jacob".upper() # JACOB (the same method; different object, different result)
A function bound to a class (other name: class method) means that a user have to bind it to a particular class — again with the same dot (.
) notation.
Examples:
A.some_class_method() # A is a class
B.some_class_method(parameters) # B is a class
From the designer point of view there are the same 3 sorts of functions, so I use the same numbers for them:
A free (unbound) function is defined out of a class:
def free_function(parameters):
...
A function bound to an object is defined inside a class, with the first parameter reserved for an object and named self
(only a convention, but a very strong one):
class A:
def bound_to_object(self, other_parameters):
...
A function bound to a class is defined inside a class, with the first parameter reserved for a class and named cls
(only a convention, but a very strong one) and with the @classmethod
decorator just before it:
class A:
@classmethod
def bound_to_class(cls, other_parameters):
...
Edit: in the context of super
, much of below is wrong. See the comment by John Y.
super(Foo, a).bar
returns the method called bar
from the next object in the method resolution order (the MRO), in this case bound to the object a
, an instance of Foo
. If a
was left out, then the returned method would be unbound. Methods are just objects, but they can be bound or unbound.
An unbound method is a method that is not tied to an instance of a class. It doesn't receive the instance of the class as the implicit first argument.
You can still call unbound methods, but you need to pass an instance of the class explicitly as the first argument.
The following gives an example of a bound and an unbound method and how to use them.
In [1]: class Foo(object):
...: def bar(self):
...: print self
...:
In [2]: Foo.bar
Out[2]: <unbound method Foo.bar>
In [3]: a = Foo()
In [4]: a.bar
Out[4]: <bound method Foo.bar of <__main__.Foo object at 0x4433110>>
In [5]: a.bar()
<__main__.Foo object at 0x4433110>
In [6]: Foo.bar()
---------------------------------------------------------------------------
TypeError Traceback (most recent call last)
<ipython-input-6-bb3335dac614> in <module>()
----> 1 Foo.bar()
TypeError: unbound method bar() must be called with Foo instance as first argument (got nothing instead)
In [7]: Foo.bar(a)
<__main__.Foo object at 0x4433110>
"Unbound" means it will return the class, rather than an instance of the class.
Other answers (answer, answer) for your question already explained the meaning of the words bound / unbound.
So my focus is to explain only the use of an unbound proxy object returned from the
super()
function (i.e. in the case it was used only with 1 argument).
The reason for obtaining an unbound object is to bind it later.
Particularly, for the unbound object returned from the super()
function you may bind it to an appropriate object using its __get__()
method, e.g.
super(C).__get__(c) # where C is a class and c is an object
To illustrate it, let's create 3 dependent classes and 3 objects - one for every class:
class A:
def __init__(self, name):
self.name = name
def message(self, source):
print(f"From: {source}, class: A, object: {self.name}")
class B(A):
def __init__(self, name):
self.name = name
def message(self, source):
print(f"From: {source}, class: B, object: {self.name}")
class C(B):
def __init__(self, name):
self.name = name
def message(self, source):
print(f"From: {source}, class: C, object: {self.name}")
a = A("a")
b = B("b")
c = C("c")
Now in the interactive console, at first for understanding things:
>>> super(B) # unbounded (note 'None')
<super: __main__.B, None>
>>> super(B).__get__(b) # bounded to object b (note address)
<super: __main__.B, <__main__.B at 0xa9bdac0>>
>>> b # note: the same address
'<__main__.B at 0xa9bdac0>
then — to show results of using different combinations of classes / objects
(in our case for delegating the method .message()
):
>>> super(B).__get__(b).message("super(B)")
From: super(B), class: A, object: b
>>> super(C).__get__(c).message("super(C)")
From: super(C), class: B, object: c
>>> super(B).__get__(c).message("super(C)")
From: super(C), class: A, object: c
and finally examples for binding and unbound proxy to appropriate classes:
>>> A.name = "Class A" # Preparing for it -
>>> B.name = "Class B" # creating some class attributes
>>> super(B).__get__(B).name # Proxy super(B) bounded to class B
'Class A'
>>> super(B).__get__(C).name # Proxy super(B) bounded to class C
'Class A'
>>> super(C).__get__(C).name # Proxy super(C) bounded to class C
'Class B'