Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 1 | # This contains most of the executable examples from Guido's descr |
| 2 | # tutorial, once at |
| 3 | # |
| 4 | # http://www.python.org/2.2/descrintro.html |
| 5 | # |
| 6 | # A few examples left implicit in the writeup were fleshed out, a few were |
| 7 | # skipped due to lack of interest (e.g., faking super() by hand isn't |
| 8 | # of much interest anymore), and a few were fiddled to make the output |
| 9 | # deterministic. |
| 10 | |
Benjamin Peterson | ee8712c | 2008-05-20 21:35:26 +0000 | [diff] [blame] | 11 | from test.support import sortdict |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 12 | import pprint |
| 13 | |
Tim Peters | a427a2b | 2001-10-29 22:25:45 +0000 | [diff] [blame] | 14 | class defaultdict(dict): |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 15 | def __init__(self, default=None): |
Tim Peters | a427a2b | 2001-10-29 22:25:45 +0000 | [diff] [blame] | 16 | dict.__init__(self) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 17 | self.default = default |
| 18 | |
| 19 | def __getitem__(self, key): |
| 20 | try: |
Tim Peters | a427a2b | 2001-10-29 22:25:45 +0000 | [diff] [blame] | 21 | return dict.__getitem__(self, key) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 22 | except KeyError: |
| 23 | return self.default |
| 24 | |
| 25 | def get(self, key, *args): |
| 26 | if not args: |
| 27 | args = (self.default,) |
Tim Peters | a427a2b | 2001-10-29 22:25:45 +0000 | [diff] [blame] | 28 | return dict.get(self, key, *args) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 29 | |
| 30 | def merge(self, other): |
| 31 | for key in other: |
| 32 | if key not in self: |
| 33 | self[key] = other[key] |
| 34 | |
| 35 | test_1 = """ |
| 36 | |
| 37 | Here's the new type at work: |
| 38 | |
Georg Brandl | 88fc664 | 2007-02-09 21:28:07 +0000 | [diff] [blame] | 39 | >>> print(defaultdict) # show our type |
Benjamin Peterson | ab078e9 | 2016-07-13 21:13:29 -0700 | [diff] [blame] | 40 | <class 'test.test_descrtut.defaultdict'> |
Georg Brandl | 88fc664 | 2007-02-09 21:28:07 +0000 | [diff] [blame] | 41 | >>> print(type(defaultdict)) # its metatype |
Benjamin Peterson | ab078e9 | 2016-07-13 21:13:29 -0700 | [diff] [blame] | 42 | <class 'type'> |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 43 | >>> a = defaultdict(default=0.0) # create an instance |
Georg Brandl | 88fc664 | 2007-02-09 21:28:07 +0000 | [diff] [blame] | 44 | >>> print(a) # show the instance |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 45 | {} |
Georg Brandl | 88fc664 | 2007-02-09 21:28:07 +0000 | [diff] [blame] | 46 | >>> print(type(a)) # show its type |
Benjamin Peterson | ab078e9 | 2016-07-13 21:13:29 -0700 | [diff] [blame] | 47 | <class 'test.test_descrtut.defaultdict'> |
Georg Brandl | 88fc664 | 2007-02-09 21:28:07 +0000 | [diff] [blame] | 48 | >>> print(a.__class__) # show its class |
Benjamin Peterson | ab078e9 | 2016-07-13 21:13:29 -0700 | [diff] [blame] | 49 | <class 'test.test_descrtut.defaultdict'> |
Georg Brandl | 88fc664 | 2007-02-09 21:28:07 +0000 | [diff] [blame] | 50 | >>> print(type(a) is a.__class__) # its type is its class |
Guido van Rossum | 77f6a65 | 2002-04-03 22:41:51 +0000 | [diff] [blame] | 51 | True |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 52 | >>> a[1] = 3.25 # modify the instance |
Georg Brandl | 88fc664 | 2007-02-09 21:28:07 +0000 | [diff] [blame] | 53 | >>> print(a) # show the new value |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 54 | {1: 3.25} |
Georg Brandl | 88fc664 | 2007-02-09 21:28:07 +0000 | [diff] [blame] | 55 | >>> print(a[1]) # show the new item |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 56 | 3.25 |
Mark Dickinson | 934896d | 2009-02-21 20:59:32 +0000 | [diff] [blame] | 57 | >>> print(a[0]) # a non-existent item |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 58 | 0.0 |
Tim Peters | a427a2b | 2001-10-29 22:25:45 +0000 | [diff] [blame] | 59 | >>> a.merge({1:100, 2:200}) # use a dict method |
Georg Brandl | 88fc664 | 2007-02-09 21:28:07 +0000 | [diff] [blame] | 60 | >>> print(sortdict(a)) # show the result |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 61 | {1: 3.25, 2: 200} |
| 62 | >>> |
| 63 | |
| 64 | We can also use the new type in contexts where classic only allows "real" |
| 65 | dictionaries, such as the locals/globals dictionaries for the exec |
| 66 | statement or the built-in function eval(): |
| 67 | |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 68 | >>> print(sorted(a.keys())) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 69 | [1, 2] |
Georg Brandl | 88fc664 | 2007-02-09 21:28:07 +0000 | [diff] [blame] | 70 | >>> a['print'] = print # need the print function here |
| 71 | >>> exec("x = 3; print(x)", a) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 72 | 3 |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 73 | >>> print(sorted(a.keys(), key=lambda x: (str(type(x)), x))) |
Georg Brandl | 88fc664 | 2007-02-09 21:28:07 +0000 | [diff] [blame] | 74 | [1, 2, '__builtins__', 'print', 'x'] |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 75 | >>> print(a['x']) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 76 | 3 |
| 77 | >>> |
| 78 | |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 79 | Now I'll show that defaultdict instances have dynamic instance variables, |
| 80 | just like classic classes: |
| 81 | |
| 82 | >>> a.default = -1 |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 83 | >>> print(a["noway"]) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 84 | -1 |
| 85 | >>> a.default = -1000 |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 86 | >>> print(a["noway"]) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 87 | -1000 |
Tim Peters | 5d2b77c | 2001-09-03 05:47:38 +0000 | [diff] [blame] | 88 | >>> 'default' in dir(a) |
Guido van Rossum | 77f6a65 | 2002-04-03 22:41:51 +0000 | [diff] [blame] | 89 | True |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 90 | >>> a.x1 = 100 |
| 91 | >>> a.x2 = 200 |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 92 | >>> print(a.x1) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 93 | 100 |
Tim Peters | 5d2b77c | 2001-09-03 05:47:38 +0000 | [diff] [blame] | 94 | >>> d = dir(a) |
| 95 | >>> 'default' in d and 'x1' in d and 'x2' in d |
Guido van Rossum | 77f6a65 | 2002-04-03 22:41:51 +0000 | [diff] [blame] | 96 | True |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 97 | >>> print(sortdict(a.__dict__)) |
Tim Peters | e2052ab | 2003-02-18 16:54:41 +0000 | [diff] [blame] | 98 | {'default': -1000, 'x1': 100, 'x2': 200} |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 99 | >>> |
| 100 | """ |
| 101 | |
Tim Peters | a427a2b | 2001-10-29 22:25:45 +0000 | [diff] [blame] | 102 | class defaultdict2(dict): |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 103 | __slots__ = ['default'] |
| 104 | |
| 105 | def __init__(self, default=None): |
Tim Peters | a427a2b | 2001-10-29 22:25:45 +0000 | [diff] [blame] | 106 | dict.__init__(self) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 107 | self.default = default |
| 108 | |
| 109 | def __getitem__(self, key): |
| 110 | try: |
Tim Peters | a427a2b | 2001-10-29 22:25:45 +0000 | [diff] [blame] | 111 | return dict.__getitem__(self, key) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 112 | except KeyError: |
| 113 | return self.default |
| 114 | |
| 115 | def get(self, key, *args): |
| 116 | if not args: |
| 117 | args = (self.default,) |
Tim Peters | a427a2b | 2001-10-29 22:25:45 +0000 | [diff] [blame] | 118 | return dict.get(self, key, *args) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 119 | |
| 120 | def merge(self, other): |
| 121 | for key in other: |
| 122 | if key not in self: |
| 123 | self[key] = other[key] |
| 124 | |
| 125 | test_2 = """ |
| 126 | |
| 127 | The __slots__ declaration takes a list of instance variables, and reserves |
| 128 | space for exactly these in the instance. When __slots__ is used, other |
| 129 | instance variables cannot be assigned to: |
| 130 | |
| 131 | >>> a = defaultdict2(default=0.0) |
| 132 | >>> a[1] |
| 133 | 0.0 |
| 134 | >>> a.default = -1 |
| 135 | >>> a[1] |
| 136 | -1 |
| 137 | >>> a.x1 = 1 |
| 138 | Traceback (most recent call last): |
| 139 | File "<stdin>", line 1, in ? |
| 140 | AttributeError: 'defaultdict2' object has no attribute 'x1' |
| 141 | >>> |
| 142 | |
| 143 | """ |
| 144 | |
| 145 | test_3 = """ |
| 146 | |
| 147 | Introspecting instances of built-in types |
| 148 | |
| 149 | For instance of built-in types, x.__class__ is now the same as type(x): |
| 150 | |
| 151 | >>> type([]) |
Benjamin Peterson | ab078e9 | 2016-07-13 21:13:29 -0700 | [diff] [blame] | 152 | <class 'list'> |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 153 | >>> [].__class__ |
Benjamin Peterson | ab078e9 | 2016-07-13 21:13:29 -0700 | [diff] [blame] | 154 | <class 'list'> |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 155 | >>> list |
Benjamin Peterson | ab078e9 | 2016-07-13 21:13:29 -0700 | [diff] [blame] | 156 | <class 'list'> |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 157 | >>> isinstance([], list) |
Guido van Rossum | 77f6a65 | 2002-04-03 22:41:51 +0000 | [diff] [blame] | 158 | True |
Tim Peters | a427a2b | 2001-10-29 22:25:45 +0000 | [diff] [blame] | 159 | >>> isinstance([], dict) |
Guido van Rossum | 77f6a65 | 2002-04-03 22:41:51 +0000 | [diff] [blame] | 160 | False |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 161 | >>> isinstance([], object) |
Guido van Rossum | 77f6a65 | 2002-04-03 22:41:51 +0000 | [diff] [blame] | 162 | True |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 163 | >>> |
| 164 | |
Neal Norwitz | 8dfc4a9 | 2007-08-11 06:39:53 +0000 | [diff] [blame] | 165 | You can get the information from the list type: |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 166 | |
| 167 | >>> pprint.pprint(dir(list)) # like list.__dict__.keys(), but sorted |
| 168 | ['__add__', |
| 169 | '__class__', |
Guido van Rossum | 48b069a | 2020-04-07 09:50:06 -0700 | [diff] [blame] | 170 | '__class_getitem__', |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 171 | '__contains__', |
| 172 | '__delattr__', |
| 173 | '__delitem__', |
Benjamin Peterson | 82b00c1 | 2011-05-24 11:09:06 -0500 | [diff] [blame] | 174 | '__dir__', |
Tim Peters | 8044055 | 2002-02-19 04:25:19 +0000 | [diff] [blame] | 175 | '__doc__', |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 176 | '__eq__', |
Eric Smith | 8c66326 | 2007-08-25 02:26:07 +0000 | [diff] [blame] | 177 | '__format__', |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 178 | '__ge__', |
Guido van Rossum | 867a8d2 | 2001-09-21 19:29:08 +0000 | [diff] [blame] | 179 | '__getattribute__', |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 180 | '__getitem__', |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 181 | '__gt__', |
| 182 | '__hash__', |
| 183 | '__iadd__', |
| 184 | '__imul__', |
| 185 | '__init__', |
Nick Coghlan | d78448e | 2016-07-30 16:26:03 +1000 | [diff] [blame] | 186 | '__init_subclass__', |
Raymond Hettinger | 14bd6de | 2002-05-31 21:40:38 +0000 | [diff] [blame] | 187 | '__iter__', |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 188 | '__le__', |
| 189 | '__len__', |
| 190 | '__lt__', |
| 191 | '__mul__', |
| 192 | '__ne__', |
| 193 | '__new__', |
Guido van Rossum | 3926a63 | 2001-09-25 16:25:58 +0000 | [diff] [blame] | 194 | '__reduce__', |
Guido van Rossum | c53f009 | 2003-02-18 22:05:12 +0000 | [diff] [blame] | 195 | '__reduce_ex__', |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 196 | '__repr__', |
Raymond Hettinger | af28e4b | 2003-11-08 12:39:53 +0000 | [diff] [blame] | 197 | '__reversed__', |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 198 | '__rmul__', |
| 199 | '__setattr__', |
| 200 | '__setitem__', |
Martin v. Löwis | 00709aa | 2008-06-04 14:18:43 +0000 | [diff] [blame] | 201 | '__sizeof__', |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 202 | '__str__', |
Christian Heimes | 9e7f1d2 | 2008-02-28 12:27:11 +0000 | [diff] [blame] | 203 | '__subclasshook__', |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 204 | 'append', |
Eli Bendersky | cbbaa96 | 2011-02-25 05:47:53 +0000 | [diff] [blame] | 205 | 'clear', |
| 206 | 'copy', |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 207 | 'count', |
| 208 | 'extend', |
| 209 | 'index', |
| 210 | 'insert', |
| 211 | 'pop', |
| 212 | 'remove', |
| 213 | 'reverse', |
Raymond Hettinger | 64958a1 | 2003-12-17 20:43:33 +0000 | [diff] [blame] | 214 | 'sort'] |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 215 | |
| 216 | The new introspection API gives more information than the old one: in |
| 217 | addition to the regular methods, it also shows the methods that are |
| 218 | normally invoked through special notations, e.g. __iadd__ (+=), __len__ |
| 219 | (len), __ne__ (!=). You can invoke any method from this list directly: |
| 220 | |
| 221 | >>> a = ['tic', 'tac'] |
| 222 | >>> list.__len__(a) # same as len(a) |
| 223 | 2 |
| 224 | >>> a.__len__() # ditto |
| 225 | 2 |
| 226 | >>> list.append(a, 'toe') # same as a.append('toe') |
| 227 | >>> a |
| 228 | ['tic', 'tac', 'toe'] |
| 229 | >>> |
| 230 | |
| 231 | This is just like it is for user-defined classes. |
| 232 | """ |
| 233 | |
| 234 | test_4 = """ |
| 235 | |
| 236 | Static methods and class methods |
| 237 | |
| 238 | The new introspection API makes it possible to add static methods and class |
| 239 | methods. Static methods are easy to describe: they behave pretty much like |
| 240 | static methods in C++ or Java. Here's an example: |
| 241 | |
| 242 | >>> class C: |
Guido van Rossum | d8faa36 | 2007-04-27 19:54:29 +0000 | [diff] [blame] | 243 | ... |
Guido van Rossum | 5a8a037 | 2005-01-16 00:25:31 +0000 | [diff] [blame] | 244 | ... @staticmethod |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 245 | ... def foo(x, y): |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 246 | ... print("staticmethod", x, y) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 247 | |
| 248 | >>> C.foo(1, 2) |
| 249 | staticmethod 1 2 |
| 250 | >>> c = C() |
| 251 | >>> c.foo(1, 2) |
| 252 | staticmethod 1 2 |
| 253 | |
| 254 | Class methods use a similar pattern to declare methods that receive an |
| 255 | implicit first argument that is the *class* for which they are invoked. |
| 256 | |
| 257 | >>> class C: |
Guido van Rossum | 5a8a037 | 2005-01-16 00:25:31 +0000 | [diff] [blame] | 258 | ... @classmethod |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 259 | ... def foo(cls, y): |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 260 | ... print("classmethod", cls, y) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 261 | |
| 262 | >>> C.foo(1) |
Benjamin Peterson | ab078e9 | 2016-07-13 21:13:29 -0700 | [diff] [blame] | 263 | classmethod <class 'test.test_descrtut.C'> 1 |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 264 | >>> c = C() |
| 265 | >>> c.foo(1) |
Benjamin Peterson | ab078e9 | 2016-07-13 21:13:29 -0700 | [diff] [blame] | 266 | classmethod <class 'test.test_descrtut.C'> 1 |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 267 | |
| 268 | >>> class D(C): |
| 269 | ... pass |
| 270 | |
| 271 | >>> D.foo(1) |
Benjamin Peterson | ab078e9 | 2016-07-13 21:13:29 -0700 | [diff] [blame] | 272 | classmethod <class 'test.test_descrtut.D'> 1 |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 273 | >>> d = D() |
| 274 | >>> d.foo(1) |
Benjamin Peterson | ab078e9 | 2016-07-13 21:13:29 -0700 | [diff] [blame] | 275 | classmethod <class 'test.test_descrtut.D'> 1 |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 276 | |
| 277 | This prints "classmethod __main__.D 1" both times; in other words, the |
| 278 | class passed as the first argument of foo() is the class involved in the |
| 279 | call, not the class involved in the definition of foo(). |
| 280 | |
| 281 | But notice this: |
| 282 | |
| 283 | >>> class E(C): |
Guido van Rossum | 5a8a037 | 2005-01-16 00:25:31 +0000 | [diff] [blame] | 284 | ... @classmethod |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 285 | ... def foo(cls, y): # override C.foo |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 286 | ... print("E.foo() called") |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 287 | ... C.foo(y) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 288 | |
| 289 | >>> E.foo(1) |
| 290 | E.foo() called |
Benjamin Peterson | ab078e9 | 2016-07-13 21:13:29 -0700 | [diff] [blame] | 291 | classmethod <class 'test.test_descrtut.C'> 1 |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 292 | >>> e = E() |
| 293 | >>> e.foo(1) |
| 294 | E.foo() called |
Benjamin Peterson | ab078e9 | 2016-07-13 21:13:29 -0700 | [diff] [blame] | 295 | classmethod <class 'test.test_descrtut.C'> 1 |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 296 | |
| 297 | In this example, the call to C.foo() from E.foo() will see class C as its |
| 298 | first argument, not class E. This is to be expected, since the call |
| 299 | specifies the class C. But it stresses the difference between these class |
| 300 | methods and methods defined in metaclasses (where an upcall to a metamethod |
| 301 | would pass the target class as an explicit first argument). |
| 302 | """ |
| 303 | |
| 304 | test_5 = """ |
| 305 | |
| 306 | Attributes defined by get/set methods |
| 307 | |
| 308 | |
Guido van Rossum | 8bce4ac | 2001-09-06 21:56:42 +0000 | [diff] [blame] | 309 | >>> class property(object): |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 310 | ... |
| 311 | ... def __init__(self, get, set=None): |
| 312 | ... self.__get = get |
| 313 | ... self.__set = set |
| 314 | ... |
| 315 | ... def __get__(self, inst, type=None): |
| 316 | ... return self.__get(inst) |
| 317 | ... |
| 318 | ... def __set__(self, inst, value): |
| 319 | ... if self.__set is None: |
Collin Winter | 3add4d7 | 2007-08-29 23:37:32 +0000 | [diff] [blame] | 320 | ... raise AttributeError("this attribute is read-only") |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 321 | ... return self.__set(inst, value) |
| 322 | |
| 323 | Now let's define a class with an attribute x defined by a pair of methods, |
Terry Jan Reedy | c30b7b1 | 2013-03-11 17:57:08 -0400 | [diff] [blame] | 324 | getx() and setx(): |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 325 | |
| 326 | >>> class C(object): |
| 327 | ... |
| 328 | ... def __init__(self): |
| 329 | ... self.__x = 0 |
| 330 | ... |
| 331 | ... def getx(self): |
| 332 | ... return self.__x |
| 333 | ... |
| 334 | ... def setx(self, x): |
| 335 | ... if x < 0: x = 0 |
| 336 | ... self.__x = x |
| 337 | ... |
Guido van Rossum | 8bce4ac | 2001-09-06 21:56:42 +0000 | [diff] [blame] | 338 | ... x = property(getx, setx) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 339 | |
| 340 | Here's a small demonstration: |
| 341 | |
| 342 | >>> a = C() |
| 343 | >>> a.x = 10 |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 344 | >>> print(a.x) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 345 | 10 |
| 346 | >>> a.x = -10 |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 347 | >>> print(a.x) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 348 | 0 |
| 349 | >>> |
| 350 | |
Guido van Rossum | 8bce4ac | 2001-09-06 21:56:42 +0000 | [diff] [blame] | 351 | Hmm -- property is builtin now, so let's try it that way too. |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 352 | |
Guido van Rossum | 8bce4ac | 2001-09-06 21:56:42 +0000 | [diff] [blame] | 353 | >>> del property # unmask the builtin |
| 354 | >>> property |
Benjamin Peterson | ab078e9 | 2016-07-13 21:13:29 -0700 | [diff] [blame] | 355 | <class 'property'> |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 356 | |
| 357 | >>> class C(object): |
| 358 | ... def __init__(self): |
| 359 | ... self.__x = 0 |
| 360 | ... def getx(self): |
| 361 | ... return self.__x |
| 362 | ... def setx(self, x): |
| 363 | ... if x < 0: x = 0 |
| 364 | ... self.__x = x |
Guido van Rossum | 8bce4ac | 2001-09-06 21:56:42 +0000 | [diff] [blame] | 365 | ... x = property(getx, setx) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 366 | |
| 367 | |
| 368 | >>> a = C() |
| 369 | >>> a.x = 10 |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 370 | >>> print(a.x) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 371 | 10 |
| 372 | >>> a.x = -10 |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 373 | >>> print(a.x) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 374 | 0 |
| 375 | >>> |
| 376 | """ |
| 377 | |
| 378 | test_6 = """ |
| 379 | |
| 380 | Method resolution order |
| 381 | |
| 382 | This example is implicit in the writeup. |
| 383 | |
Thomas Wouters | 28bc768 | 2006-04-15 09:03:16 +0000 | [diff] [blame] | 384 | >>> class A: # implicit new-style class |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 385 | ... def save(self): |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 386 | ... print("called A.save()") |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 387 | >>> class B(A): |
| 388 | ... pass |
| 389 | >>> class C(A): |
| 390 | ... def save(self): |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 391 | ... print("called C.save()") |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 392 | >>> class D(B, C): |
| 393 | ... pass |
| 394 | |
| 395 | >>> D().save() |
Thomas Wouters | 28bc768 | 2006-04-15 09:03:16 +0000 | [diff] [blame] | 396 | called C.save() |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 397 | |
Thomas Wouters | 28bc768 | 2006-04-15 09:03:16 +0000 | [diff] [blame] | 398 | >>> class A(object): # explicit new-style class |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 399 | ... def save(self): |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 400 | ... print("called A.save()") |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 401 | >>> class B(A): |
| 402 | ... pass |
| 403 | >>> class C(A): |
| 404 | ... def save(self): |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 405 | ... print("called C.save()") |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 406 | >>> class D(B, C): |
| 407 | ... pass |
| 408 | |
| 409 | >>> D().save() |
| 410 | called C.save() |
| 411 | """ |
| 412 | |
| 413 | class A(object): |
| 414 | def m(self): |
| 415 | return "A" |
| 416 | |
| 417 | class B(A): |
| 418 | def m(self): |
| 419 | return "B" + super(B, self).m() |
| 420 | |
| 421 | class C(A): |
| 422 | def m(self): |
| 423 | return "C" + super(C, self).m() |
| 424 | |
| 425 | class D(C, B): |
| 426 | def m(self): |
| 427 | return "D" + super(D, self).m() |
| 428 | |
| 429 | |
| 430 | test_7 = """ |
| 431 | |
| 432 | Cooperative methods and "super" |
| 433 | |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 434 | >>> print(D().m()) # "DCBA" |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 435 | DCBA |
| 436 | """ |
| 437 | |
| 438 | test_8 = """ |
| 439 | |
| 440 | Backwards incompatibilities |
| 441 | |
| 442 | >>> class A: |
| 443 | ... def foo(self): |
Guido van Rossum | 7131f84 | 2007-02-09 20:13:25 +0000 | [diff] [blame] | 444 | ... print("called A.foo()") |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 445 | |
| 446 | >>> class B(A): |
| 447 | ... pass |
| 448 | |
| 449 | >>> class C(A): |
| 450 | ... def foo(self): |
| 451 | ... B.foo(self) |
| 452 | |
| 453 | >>> C().foo() |
Christian Heimes | 4a22b5d | 2007-11-25 09:39:14 +0000 | [diff] [blame] | 454 | called A.foo() |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 455 | |
| 456 | >>> class C(A): |
| 457 | ... def foo(self): |
| 458 | ... A.foo(self) |
| 459 | >>> C().foo() |
| 460 | called A.foo() |
| 461 | """ |
| 462 | |
| 463 | __test__ = {"tut1": test_1, |
| 464 | "tut2": test_2, |
| 465 | "tut3": test_3, |
| 466 | "tut4": test_4, |
| 467 | "tut5": test_5, |
| 468 | "tut6": test_6, |
| 469 | "tut7": test_7, |
| 470 | "tut8": test_8} |
| 471 | |
| 472 | # Magic test name that regrtest.py invokes *after* importing this module. |
| 473 | # This worms around a bootstrap problem. |
| 474 | # Note that doctest and regrtest both look in sys.argv for a "-v" argument, |
| 475 | # so this works as expected in both ways of running regrtest. |
Tim Peters | a0a6222 | 2001-09-09 06:12:01 +0000 | [diff] [blame] | 476 | def test_main(verbose=None): |
| 477 | # Obscure: import this module as test.test_descrtut instead of as |
| 478 | # plain test_descrtut because the name of this module works its way |
| 479 | # into the doctest examples, and unless the full test.test_descrtut |
| 480 | # business is used the name can change depending on how the test is |
| 481 | # invoked. |
Benjamin Peterson | ee8712c | 2008-05-20 21:35:26 +0000 | [diff] [blame] | 482 | from test import support, test_descrtut |
Benjamin Peterson | ab078e9 | 2016-07-13 21:13:29 -0700 | [diff] [blame] | 483 | support.run_doctest(test_descrtut, verbose) |
Tim Peters | 95c99e5 | 2001-09-03 01:24:30 +0000 | [diff] [blame] | 484 | |
| 485 | # This part isn't needed for regrtest, but for running the test directly. |
| 486 | if __name__ == "__main__": |
Tim Peters | a0a6222 | 2001-09-09 06:12:01 +0000 | [diff] [blame] | 487 | test_main(1) |