timeit and its default_timer completely disagree

前端 未结 1 1607
Happy的楠姐
Happy的楠姐 2021-01-30 08:30

I benchmarked these two functions (they unzip pairs back into source lists, came from here):

n = 10**7
a = list(range(n))
b = list(range(n))
pairs = list(zip(a, b         


        
1条回答
  •  遥遥无期
    2021-01-30 09:18

    As Martijn commented, the difference is Python's garbage collection, which timeit.timeit disables during its run. And zip creates 10 million iterator objects, one for each of the 10 million iterables it's given.

    So, garbage-collecting 10 million objects simply takes a lot of time, right? Mystery solved!

    Well... no. That's not really what happens, and it's way more interesting than that. And there's a lesson to be learned to make such code faster in real life.

    Python's main way to discard objects no longer needed is reference counting. The garbage collector, which is being disabled here, is for reference cycles, which the reference counting won't catch. And there aren't any cycles here, so it's all discarded by reference counting and the garbage collector doesn't actually collect any garbage.

    Let's look at a few things. First, let's reproduce the much faster time by disabling the garbage collector ourselves.

    Common setup code (all further blocks of code should be run directly after this in a fresh run, don't combine them):

    import gc
    from timeit import default_timer as timer
    
    n = 10**7
    a = list(range(n))
    b = list(range(n))
    pairs = list(zip(a, b))
    

    Timing with garbage collection enabled (the default):

    t0 = timer()
    a[:], b[:] = zip(*pairs)
    t1 = timer()
    print(t1 - t0)
    

    I ran it three times, took 7.09, 7.03 and 7.09 seconds.

    Timing with garbage collection disabled:

    t0 = timer()
    gc.disable()
    a[:], b[:] = zip(*pairs)
    gc.enable()
    t1 = timer()
    print(t1 - t0)
    

    Took 0.96, 1.02 and 0.99 seconds.

    So now we know it's indeed the garbage collection that somehow takes most of the time, even though it's not collecting anything.

    Here's something interesting: Already just the creation of the zip iterator is responsible for most of the time:

    t0 = timer()
    z = zip(*pairs)
    t1 = timer()
    print(t1 - t0)
    

    That took 6.52, 6.51 and 6.50 seconds.

    Note that I kept the zip iterator in a variable, so there isn't even anything to discard yet, neither by reference counting nor by garbage collecting!

    What?! Where does the time go, then?

    Well... as I said, there are no reference cycles, so the garbage collector won't actually collect any garbage. But the garbage collector doesn't know that! In order to figure that out, it needs to check!

    Since the iterators could become part of a reference cycle, they're registered for garbage collection tracking. Let's see how many more objects get tracked due to the zip creation (doing this just after the common setup code):

    gc.collect()
    tracked_before = len(gc.get_objects())
    z = zip(*pairs)
    print(len(gc.get_objects()) - tracked_before)
    

    The output: 10000003 new objects tracked. I believe that's the zip object itself, its internal tuple to hold the iterators, its internal result holder tuple, and the 10 million iterators.

    Ok, so the garbage collector tracks all these objects. But what does that mean? Well, every now and then, after a certain number of new object creations, the collector goes through the tracked objects to see whether some are garbage and can be discarded. The collector keeps three "generations" of tracked objects. New objects go into generation 0. If they survive a collection run there, they're moved into generation 1. If they survive a collection there, they're moved into generation 2. If they survive further collection runs there, they remain in generation 2. Let's check the generations before and after:

    gc.collect()
    print('collections:', [stats['collections'] for stats in gc.get_stats()])
    print('objects:', [len(gc.get_objects(i)) for i in range(3)])
    z = zip(*pairs)
    print('collections:', [stats['collections'] for stats in gc.get_stats()])
    print('objects:', [len(gc.get_objects(i)) for i in range(3)])
    

    Output (each line shows values for the three generations):

    collections: [13111, 1191, 2]
    objects: [17, 0, 13540]
    collections: [26171, 2378, 20]
    objects: [317, 2103, 10011140]
    

    The 10011140 shows that most of the 10 million iterators were not just registered for tracking, but are already in generation 2. So they were part of at least two garbage collection runs. And the number of generation 2 collections went up from 2 to 20, so our millions of iterators were part of up to 20 garbage collection runs (two to get into generation 2, and up to 18 more while already in generation 2). We can also register a callback to count more precisely:

    checks = 0
    def count(phase, info):
        if phase == 'start':
            global checks
            checks += len(gc.get_objects(info['generation']))
    
    gc.callbacks.append(count)
    z = zip(*pairs)
    gc.callbacks.remove(count)
    print(checks)
    

    That told me 63,891,314 checks total (i.e., on average, each iterator was part of over 6 garbage collection runs). That's a lot of work. And all this just to create the zip iterator, before even using it.

    Meanwhile, the loop

    for i, (a[i], b[i]) in enumerate(pairs):
        pass
    

    creates almost no new objects at all. Let's check how much tracking enumerate causes:

    gc.collect()
    tracked_before = len(gc.get_objects())
    e = enumerate(pairs)
    print(len(gc.get_objects()) - tracked_before)
    

    Output: 3 new objects tracked (the enumerate iterator object itself, the single iterator it creates for iterating over pairs, and the result tuple it'll use (code here)).

    I'd say that answers the question "Why do the timings totally differ like that?". The zip solution creates millions of objects that go through multiple garbage collection runs, while the loop solution doesn't. So disabling the garbage collector helps the zip solution tremendously, while the loop solution doesn't care.

    Now about the second question: "Which timing method should I believe?". Here's what the documentation has to say about it (emphasis mine):

    By default, timeit() temporarily turns off garbage collection during the timing. The advantage of this approach is that it makes independent timings more comparable. The disadvantage is that GC may be an important component of the performance of the function being measured. If so, GC can be re-enabled as the first statement in the setup string. For example:

    timeit.Timer('for i in range(10): oct(i)', 'gc.enable()').timeit()
    

    In our case here, the cost of garbage collection doesn't stem from some other unrelated code. It's directly caused by the zip call. And you do pay this price in reality, when you run that. So in this case, I do consider it an "important component of the performance of the function being measured". To directly answer the question as asked: Here I'd believe the default_timer method, not the timeit method. Or put differently: Here the timeit method should be used with enabling garbage collection as suggested in the documentatiion.

    Or... alternatively, we could actually disable garbage collection as part of the solution (not just for benchmarking):

    def f1(a, b, pairs):
        gc.disable()
        a[:], b[:] = zip(*pairs)
        gc.enable()
    

    But is that a good idea? Here's what the gc documentation says:

    Since the collector supplements the reference counting already used in Python, you can disable the collector if you are sure your program does not create reference cycles.

    Sounds like it's an ok thing to do. But I'm not sure I don't create reference cycles elsewhere in my program, so I finish with gc.enable() to turn garbage collection back on after I'm done. At that point, all those temporary objects have already been discarded thanks to reference counting. So all I'm doing is avoiding lots of pointless garbage collection checks. I find this a valuable lesson and I might actually do that in the future, if I know I only temporarily create a lot of objects.

    Finally, I highly recommend reading the gc module documentation and the Design of CPython’s Garbage Collector in Python's developer guide. Most of it is easy to understand, and I found it quite interesting and enlightening.

    0 讨论(0)
提交回复
热议问题