Thanks to visit codestin.com
Credit goes to github.com

Skip to content

Commit dc879f0

Browse files
committed
Forward port r70470 and r70473 for OrderedDict to use a doubly linked list.
1 parent 6cf17aa commit dc879f0

3 files changed

Lines changed: 44 additions & 17 deletions

File tree

Doc/library/collections.rst

Lines changed: 10 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -836,8 +836,11 @@ the items are returned in the order their keys were first added.
836836

837837
.. versionadded:: 3.1
838838

839-
The :meth:`popitem` method for ordered dictionaries returns and removes the
840-
last added entry. The key/value pairs are returned in LIFO order.
839+
.. method:: OrderedDict.popitem(last=True)
840+
841+
The :meth:`popitem` method for ordered dictionaries returns and removes
842+
a (key, value) pair. The pairs are returned in LIFO order if *last* is
843+
true or FIFO order if false.
841844

842845
Equality tests between :class:`OrderedDict` objects are order-sensitive
843846
and are implemented as ``list(od1.items())==list(od2.items())``.
@@ -846,6 +849,11 @@ Equality tests between :class:`OrderedDict` objects and other
846849
This allows :class:`OrderedDict` objects to be substituted anywhere a
847850
regular dictionary is used.
848851

852+
.. seealso::
853+
854+
`Equivalent OrderedDict recipe <http://code.activestate.com/recipes/576693/>`_
855+
that runs on Python 2.4 or later.
856+
849857

850858
:class:`UserDict` objects
851859
-------------------------

Lib/collections.py

Lines changed: 27 additions & 15 deletions
Original file line numberDiff line numberDiff line change
@@ -23,45 +23,57 @@ def __init__(self, *args, **kwds):
2323
if len(args) > 1:
2424
raise TypeError('expected at most 1 arguments, got %d' % len(args))
2525
try:
26-
self.__keys
26+
self.__end
2727
except AttributeError:
28-
# Note the underlying data structure for this class is likely to
29-
# change in the future. Do not rely on it or access it directly.
30-
self.__keys = deque()
28+
self.clear()
3129
self.update(*args, **kwds)
3230

3331
def clear(self):
34-
self.__keys.clear()
32+
self.__end = end = []
33+
end += [None, end, end] # sentinel node for doubly linked list
34+
self.__map = {} # key --> [key, prev, next]
3535
dict.clear(self)
3636

3737
def __setitem__(self, key, value):
3838
if key not in self:
39-
self.__keys.append(key)
39+
end = self.__end
40+
curr = end[1]
41+
curr[2] = end[1] = self.__map[key] = [key, curr, end]
4042
dict.__setitem__(self, key, value)
4143

4244
def __delitem__(self, key):
4345
dict.__delitem__(self, key)
44-
self.__keys.remove(key)
46+
key, prev, next = self.__map.pop(key)
47+
prev[2] = next
48+
next[1] = prev
4549

4650
def __iter__(self):
47-
return iter(self.__keys)
51+
end = self.__end
52+
curr = end[2]
53+
while curr is not end:
54+
yield curr[0]
55+
curr = curr[2]
4856

4957
def __reversed__(self):
50-
return reversed(self.__keys)
58+
end = self.__end
59+
curr = end[1]
60+
while curr is not end:
61+
yield curr[0]
62+
curr = curr[1]
5163

52-
def popitem(self):
64+
def popitem(self, last=True):
5365
if not self:
5466
raise KeyError('dictionary is empty')
55-
key = self.__keys.pop()
56-
value = dict.pop(self, key)
67+
key = next(reversed(self)) if last else next(iter(self))
68+
value = self.pop(key)
5769
return key, value
5870

5971
def __reduce__(self):
6072
items = [[k, self[k]] for k in self]
61-
tmp = self.__keys
62-
del self.__keys
73+
tmp = self.__map, self.__end
74+
del self.__map, self.__end
6375
inst_dict = vars(self).copy()
64-
self.__keys = tmp
76+
self.__map, self.__end = tmp
6577
if inst_dict:
6678
return (self.__class__, (items,), inst_dict)
6779
return self.__class__, (items,)

Lib/test/test_collections.py

Lines changed: 7 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -770,12 +770,19 @@ def test_reinsert(self):
770770
class GeneralMappingTests(mapping_tests.BasicTestMappingProtocol):
771771
type2test = OrderedDict
772772

773+
def test_popitem(self):
774+
d = self._empty_mapping()
775+
self.assertRaises(KeyError, d.popitem)
776+
773777
class MyOrderedDict(OrderedDict):
774778
pass
775779

776780
class SubclassMappingTests(mapping_tests.BasicTestMappingProtocol):
777781
type2test = MyOrderedDict
778782

783+
def test_popitem(self):
784+
d = self._empty_mapping()
785+
self.assertRaises(KeyError, d.popitem)
779786

780787

781788
import doctest, collections

0 commit comments

Comments
 (0)