Generator using item[n-1] + item[n] memory

Peter Otten __peter__ at web.de
Sat Feb 15 06:27:19 EST 2014


Chris Angelico wrote:
> On Sat, Feb 15, 2014 at 6:27 PM, Ian Kelly <ian.g.kelly at gmail.com> wrote:
>> On Fri, Feb 14, 2014 at 8:31 PM, Nick Timkovich <prometheus235 at gmail.com>
>> wrote:
>>> OK, now the trick; adding `data = None` inside the generator works, but
>>> in my actual code I wrap my generator inside of `enumerate()`, which
>>> seems to
>>> obviate the "fix". Can I get it to play nice or am I forced to count
>>> manually. Is that a feature?
>>>> Yeah, looks like enumerate also doesn't release its reference to the
>> previous object until after it gets the next one. You'll just have to
>> make do without.
>> You could write your own enumerate function.
>> def enumerate(it, i=0):
> it = iter(it)
> while True:
> yield i, next(it)
> i += 1
>> That shouldn't keep any extra references around.

An alternative approach ist to yield weak refs and thus have the generator 
control the object lifetime. This doesn't work with the built-in list type 
though:
import weakref
try:
 from itertools import imap # py2
except ImportError:
 imap = map # py3
N = 0
def log_deleted(*args):
 global N
 N -= 1
 print("deleted, new N: {}".format(N))
def log_created():
 global N
 N += 1
 print("created, new N: {}".format(N))
def weakrefs(f):
 def weakrefs(*args, **kw):
 return imap(lambda x: weakref.proxy(x, log_deleted), f(*args, **kw))
 return weakrefs
class List(list):
 def __str__(self):
 s = str(self[:5])
 if len(self) > 10:
 s = s[:-1] + ", ... ]"
 return s
@weakrefs
def biggen():
 sizes = 1, 1, 10, 1, 1, 10, 10, 1, 1, 10, 10, 20, 1, 1, 20, 20, 1, 1
 for size in sizes:
 data = List([1] * int(size * 1e4))
 log_created()
 yield data
 data = None
if __name__ == "__main__":
 for i, x in enumerate(biggen()):
 print("{} {}".format(i, x))


More information about the Python-list mailing list

AltStyle によって変換されたページ (->オリジナル) /