Correctly bookkeep the size of TreeCache

pull/538/head
Erik Johnston 2016-01-29 10:44:46 +00:00
parent 766526e114
commit a30364c1f9
1 changed files with 28 additions and 3 deletions

View File

@ -21,7 +21,7 @@ class TreeCache(object):
node = self.root node = self.root
for k in key[:-1]: for k in key[:-1]:
node = node.setdefault(k, {}) node = node.setdefault(k, {})
node[key[-1]] = value node[key[-1]] = _Entry(value)
self.size += 1 self.size += 1
def get(self, key, default=None): def get(self, key, default=None):
@ -30,7 +30,7 @@ class TreeCache(object):
node = node.get(k, None) node = node.get(k, None)
if node is None: if node is None:
return default return default
return node.get(key[-1], default) return node.get(key[-1], _Entry(default)).value
def clear(self): def clear(self):
self.size = 0 self.size = 0
@ -60,8 +60,33 @@ class TreeCache(object):
break break
node_and_keys[i+1][0].pop(k) node_and_keys[i+1][0].pop(k)
self.size -= 1 popped, cnt = _strip_and_count_entires(popped)
self.size -= cnt
return popped return popped
def __len__(self): def __len__(self):
return self.size return self.size
class _Entry(object):
__slots__ = ["value"]
def __init__(self, value):
object.__setattr__(self, "value", value)
def _strip_and_count_entires(d):
"""Takes an _Entry or dict with leaves of _Entry's, and either returns the
value or a dictionary with _Entry's replaced by their values.
Also returns the count of _Entry's
"""
if isinstance(d, dict):
cnt = 0
for key, value in d.items():
v, n = _strip_and_count_entires(value)
d[key] = v
cnt += n
return d, cnt
else:
return d.value, 1