51

我正在修补 Pythonsetfrozenset集合类型。

最初,我认为它frozenset会提供比 更好的查找性能set,因为它是不可变的,因此可以利用存储项目的结构。

但是,对于以下实验,情况似乎并非如此:

import random
import time
import sys

def main(n):
    numbers = []
    for _ in xrange(n):
        numbers.append(random.randint(0, sys.maxint))
    set_ = set(numbers)
    frozenset_ = frozenset(set_)

    start = time.time()
    for number in numbers:
        number in set_
    set_duration = time.time() - start

    start = time.time()
    for number in numbers:
        number in frozenset_
    frozenset_duration = time.time() - start

    print "set      : %.3f" % set_duration
    print "frozenset: %.3f" % frozenset_duration


if __name__ == "__main__":
    n = int(sys.argv[1])
    main(n)

我使用 CPython 和 PyPy 执行了这段代码,结果如下:

> pypy set.py 100000000
set      : 6.156
frozenset: 6.166

> python set.py 100000000
set      : 16.824
frozenset: 17.248

frozenset在 CPython 和 PyPy 中,查找性能似乎实际上更慢。有人知道为什么会这样吗?我没有研究实现。

4

2 回答 2

92

和实现在很大程度上是共享的frozensetsetaset只是 afrozenset添加了变异方法,具有完全相同的哈希表实现。查看Objects/setobject.c源文件;顶级PyFrozenSet_Type定义与PySet_Type定义共享功能。

这里没有对 freezeset 的优化,因为在您测试成员资格时不需要计算项目的哈希值frozenset您用于针对集合进行测试的项目仍需要计算其哈希值,以便在集合哈希表中找到正确的插槽,以便您可以进行相等性测试。

因此,您的计时结果可能由于系统上运行的其他进程而关闭;您测量了挂钟时间,并没有禁用 Python 垃圾收集,也没有重复测试相同的东西。

尝试使用timeitmodule运行您的测试,其中一个值来自numbers集合,一个不在集合中:

import random
import sys
import timeit

numbers = [random.randrange(sys.maxsize) for _ in range(10000)]
set_ = set(numbers)
fset = frozenset(numbers)
present = random.choice(numbers)
notpresent = -1
test = 'present in s; notpresent in s'

settime = timeit.timeit(
    test,
    'from __main__ import set_ as s, present, notpresent')
fsettime = timeit.timeit(
    test,
    'from __main__ import fset as s, present, notpresent')

print('set      : {:.3f} seconds'.format(settime))
print('frozenset: {:.3f} seconds'.format(fsettime))

这将每个测试重复 100 万次并产生:

set      : 0.050 seconds
frozenset: 0.050 seconds
于 2016-04-11T17:55:19.740 回答
24

两种不同数据类型的原因不是为了性能,而是为了功能。因为frozensets 是不可变的,所以它们可以用作字典中的键。集不能用于此目的。

于 2018-07-16T14:12:44.387 回答