blob: 34ceb3a09bfcc712beff079cfd7859c8c62ecbbb [file] [log] [blame]
Guido van Rossum4acc25b2000-02-02 15:10:15 +00001"""A more or less complete user-defined wrapper around dictionary objects."""
Guido van Rossumae3b3a31993-11-30 13:43:54 +00002
3class UserDict:
Raymond Hettinger54405452002-11-22 00:07:40 +00004 def __init__(self, dict=None, **kwargs):
Raymond Hettingere4827eb2002-11-27 08:29:11 +00005 self.data = {}
Raymond Hettinger54405452002-11-22 00:07:40 +00006 if dict is not None:
Raymond Hettingere4827eb2002-11-27 08:29:11 +00007 if not hasattr(dict,'keys'):
8 dict = type({})(dict) # make mapping from a sequence
9 self.update(dict)
10 if len(kwargs):
11 self.update(kwargs)
Guido van Rossumb94cd961997-06-03 14:10:01 +000012 def __repr__(self): return repr(self.data)
13 def __cmp__(self, dict):
Guido van Rossum1697b9c1999-03-26 15:31:12 +000014 if isinstance(dict, UserDict):
Guido van Rossum45e2fbc1998-03-26 21:13:24 +000015 return cmp(self.data, dict.data)
Guido van Rossum1697b9c1999-03-26 15:31:12 +000016 else:
17 return cmp(self.data, dict)
Guido van Rossumb94cd961997-06-03 14:10:01 +000018 def __len__(self): return len(self.data)
19 def __getitem__(self, key): return self.data[key]
20 def __setitem__(self, key, item): self.data[key] = item
21 def __delitem__(self, key): del self.data[key]
Guido van Rossum1697b9c1999-03-26 15:31:12 +000022 def clear(self): self.data.clear()
Guido van Rossumb94cd961997-06-03 14:10:01 +000023 def copy(self):
Guido van Rossum1697b9c1999-03-26 15:31:12 +000024 if self.__class__ is UserDict:
25 return UserDict(self.data)
Guido van Rossum45e2fbc1998-03-26 21:13:24 +000026 import copy
Fred Drake3ce5af72001-11-05 17:40:48 +000027 data = self.data
28 try:
29 self.data = {}
30 c = copy.copy(self)
31 finally:
32 self.data = data
33 c.update(self)
34 return c
Guido van Rossumb94cd961997-06-03 14:10:01 +000035 def keys(self): return self.data.keys()
36 def items(self): return self.data.items()
Fred Drakebedebbd2001-05-03 04:54:41 +000037 def iteritems(self): return self.data.iteritems()
38 def iterkeys(self): return self.data.iterkeys()
39 def itervalues(self): return self.data.itervalues()
Guido van Rossumb94cd961997-06-03 14:10:01 +000040 def values(self): return self.data.values()
41 def has_key(self, key): return self.data.has_key(key)
Guido van Rossum1697b9c1999-03-26 15:31:12 +000042 def update(self, dict):
43 if isinstance(dict, UserDict):
44 self.data.update(dict.data)
45 elif isinstance(dict, type(self.data)):
46 self.data.update(dict)
Guido van Rossum45e2fbc1998-03-26 21:13:24 +000047 else:
Guido van Rossum1697b9c1999-03-26 15:31:12 +000048 for k, v in dict.items():
Martin v. Löwisf3b30742001-06-18 01:09:41 +000049 self[k] = v
Barry Warsawfc3e61c1997-10-06 17:50:04 +000050 def get(self, key, failobj=None):
Martin v. Löwisf3b30742001-06-18 01:09:41 +000051 if not self.has_key(key):
52 return failobj
53 return self[key]
Guido van Rossum164452c2000-08-08 16:12:54 +000054 def setdefault(self, key, failobj=None):
Martin v. Löwisf3b30742001-06-18 01:09:41 +000055 if not self.has_key(key):
56 self[key] = failobj
57 return self[key]
Guido van Rossumf4956252002-04-13 14:03:38 +000058 def pop(self, key):
59 return self.data.pop(key)
Guido van Rossum1072c382000-12-12 22:06:00 +000060 def popitem(self):
61 return self.data.popitem()
Tim Petersa3f98d62001-04-21 09:13:15 +000062 def __contains__(self, key):
63 return key in self.data
Raymond Hettingere4827eb2002-11-27 08:29:11 +000064 def fromkeys(cls, iterable, value=None):
65 d = cls()
66 for key in iterable:
67 d[key] = value
68 return d
69 fromkeys = classmethod(fromkeys)
Guido van Rossum2050b652001-08-07 17:40:42 +000070
71class IterableUserDict(UserDict):
Tim Petersa3f98d62001-04-21 09:13:15 +000072 def __iter__(self):
73 return iter(self.data)
Raymond Hettinger79947162002-11-15 06:46:14 +000074
75class DictMixin:
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +000076 # Mixin defining all dictionary methods for classes that already have
77 # a minimum dictionary interface including getitem, setitem, delitem,
78 # and keys. Without knowledge of the subclass constructor, the mixin
79 # does not define __init__() or copy(). In addition to the four base
80 # methods, progessively more efficiency comes with defining
81 # __contains__(), __iter__(), and iteritems().
Raymond Hettinger79947162002-11-15 06:46:14 +000082
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +000083 # second level definitions support higher levels
84 def __iter__(self):
85 for k in self.keys():
86 yield k
Raymond Hettinger79947162002-11-15 06:46:14 +000087 def has_key(self, key):
88 try:
89 value = self[key]
90 except KeyError:
91 return False
92 return True
93 __contains__ = has_key
Raymond Hettinger79947162002-11-15 06:46:14 +000094
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +000095 # third level takes advantage of second level definitions
Raymond Hettinger79947162002-11-15 06:46:14 +000096 def iteritems(self):
97 for k in self:
98 yield (k, self[k])
99 iterkeys = __iter__
100
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000101 # fourth level uses definitions from lower levels
Raymond Hettinger79947162002-11-15 06:46:14 +0000102 def itervalues(self):
103 for _, v in self.iteritems():
104 yield v
105 def values(self):
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000106 return [v for _, v in self.iteritems()]
Raymond Hettinger79947162002-11-15 06:46:14 +0000107 def items(self):
108 return list(self.iteritems())
109 def clear(self):
110 for key in self.keys():
111 del self[key]
112 def setdefault(self, key, default):
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000113 try:
114 return self[key]
115 except KeyError:
Raymond Hettinger79947162002-11-15 06:46:14 +0000116 self[key] = default
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000117 return default
Raymond Hettinger79947162002-11-15 06:46:14 +0000118 def pop(self, key):
119 value = self[key]
120 del self[key]
121 return value
122 def popitem(self):
123 try:
124 k, v = self.iteritems().next()
125 except StopIteration:
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000126 raise KeyError, 'container is empty'
Raymond Hettinger79947162002-11-15 06:46:14 +0000127 del self[k]
128 return (k, v)
129 def update(self, other):
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000130 # Make progressively weaker assumptions about "other"
131 if hasattr(other, 'iteritems'): # iteritems saves memory and lookups
132 for k, v in other.iteritems():
133 self[k] = v
134 elif hasattr(other, '__iter__'): # iter saves memory
135 for k in other:
136 self[k] = other[k]
137 else:
138 for k in other.keys():
139 self[k] = other[k]
Raymond Hettinger79947162002-11-15 06:46:14 +0000140 def get(self, key, default=None):
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000141 try:
Raymond Hettinger79947162002-11-15 06:46:14 +0000142 return self[key]
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000143 except KeyError:
144 return default
Raymond Hettinger79947162002-11-15 06:46:14 +0000145 def __repr__(self):
Raymond Hettinger8ddc176e2002-11-18 04:34:10 +0000146 return repr(dict(self.iteritems()))
147 def __cmp__(self, other):
148 if isinstance(other, DictMixin):
149 other = dict(other.iteritems())
150 return cmp(dict(self.iteritems()), other)
151 def __len__(self):
152 return len(self.keys())