在Python列表中快速交换元素的方法

50 投票
4 回答
65122 浏览
提问于 2025-04-16 09:09

有没有比下面这种方法更快的方式来交换Python中两个列表的元素呢?

L[a], L[b] = L[b], L[a]

还是说我得去用Cython或者Weave之类的东西?

4 个回答

0

我尝试这个方法,因为这是在列表中交换两个数字最简单的方式:

lst= [23, 65, 19, 90]
pos1 = lst.pop(0)
pos2 = lst.pop(1)
lst.append(pos1)
lst.append(pos2)
print(lst)
3

如果你能提供一个代表性的代码示例,我们就能更好地帮你比较不同的选择。顺便说一下,在我做的这个简单测试中,使用Shed Skin的速度大约提高了3倍,而使用PyPy的速度提高了10倍。

from time import time

def swap(L):
    for i in xrange(1000000):
        for b, a in enumerate(L):
            L[a], L[b] = L[b], L[a]

def main():
    start = time()
    L = list(reversed(range(100)))
    swap(L[:])
    print time() - start
    return L

if __name__ == "__main__":
    print len(main())

# for shedskin:
# shedskin -b -r -e listswap.py && make
# python -c "import listswap; print len(listswap.main())"
132

看起来Python编译器会优化这个结构中的临时元组:

代码:

import dis

def swap1():
  a=5
  b=4
  a, b = b, a

def swap2():
  a=5
  b=4
  c = a
  a = b
  b = c

print 'swap1():'
dis.dis(swap1)
print 'swap2():'
dis.dis(swap2)

输出:

swap1():
  6           0 LOAD_CONST               1 (5)
              3 STORE_FAST               0 (a)

  7           6 LOAD_CONST               2 (4)
              9 STORE_FAST               1 (b)

  8          12 LOAD_FAST                1 (b)
             15 LOAD_FAST                0 (a)
             18 ROT_TWO             
             19 STORE_FAST               0 (a)
             22 STORE_FAST               1 (b)
             25 LOAD_CONST               0 (None)
             28 RETURN_VALUE        
swap2():
 11           0 LOAD_CONST               1 (5)
              3 STORE_FAST               0 (a)

 12           6 LOAD_CONST               2 (4)
              9 STORE_FAST               1 (b)

 13          12 LOAD_FAST                0 (a)
             15 STORE_FAST               2 (c)

 14          18 LOAD_FAST                1 (b)
             21 STORE_FAST               0 (a)

 15          24 LOAD_FAST                2 (c)
             27 STORE_FAST               1 (b)
             30 LOAD_CONST               0 (None)
             33 RETURN_VALUE        

这里有两次加载,一个叫做 ROT_TWO 的操作,以及两次保存,而不是三次加载和三次保存。你很难找到更快的方法了。

撰写回答