Python交换一个数字中的两位数?

时间:2012-10-23 23:32:41

标签: python performance swap

在Python中交换数字中两位数的最快方法是什么?我被赋予了数字作为字符串,所以如果我能拥有与

一样快的东西,那就太好了
string[j] = string[j] ^ string[j+1] 
string[j+1] = string[j] ^ string[j+1]
string[j] = string[j] ^ string[j+1]

我见过的所有内容都比在C中要贵得多,并且涉及制作一个列表然后将列表转换回来或其中的一些变体。

4 个答案:

答案 0 :(得分:5)

这比你想象的要快,至少比Jon Clements目前在我的计时测试中的回答更快:

i, j = (i, j) if i < j else (j, i) # make sure i < j
s = s[:i] + s[j] + s[i+1:j] + s[i] + s[j+1:]

如果你想比较你得到的任何其他答案,这是我的试验台:

import timeit
import types

N = 10000
R = 3
SUFFIX = '_test'
SUFFIX_LEN = len(SUFFIX)

def setup():
    import random
    global s, i, j
    s = 'abcdefghijklmnopqrstuvwxyz'
    i = random.randrange(len(s))
    while True:
        j = random.randrange(len(s))
        if i != j: break

def swapchars_martineau(s, i, j):
    i, j = (i, j) if i < j else (j, i) # make sure i < j
    return s[:i] + s[j] + s[i+1:j] + s[i] + s[j+1:]

def swapchars_martineau_test():
    global s, i, j
    swapchars_martineau(s, i, j)

def swapchars_clements(text, fst, snd):
    ba = bytearray(text)
    ba[fst], ba[snd] = ba[snd], ba[fst]
    return str(ba)

def swapchars_clements_test():
    global s, i, j
    swapchars_clements(s, i, j)

# find all the functions named *SUFFIX in the global namespace
funcs = tuple(value for id,value in globals().items()
            if id.endswith(SUFFIX) and type(value) is types.FunctionType)

# run the timing tests and collect results
timings = [(f.func_name[:-SUFFIX_LEN],
            min(timeit.repeat(f, setup=setup, repeat=R, number=N))
           ) for f in funcs]
timings.sort(key=lambda x: x[1])  # sort by speed
fastest = timings[0][1]  # time fastest one took to run
longest = max(len(t[0]) for t in timings) # len of longest func name (w/o suffix)

print 'fastest to slowest *_test() function timings:\n' \
      ' {:,d} chars, {:,d} timeit calls, best of {:d}\n'.format(len(s), N, R)

def times_slower(speed, fastest):
    return speed/fastest - 1.0

for i in timings:
    print "{0:>{width}}{suffix}() : {1:.4f} ({2:.2f} times slower)".format(
                i[0], i[1], times_slower(i[1], fastest), width=longest, suffix=SUFFIX)

<强>附录:

对于以字符串形式给出正十进制数字交换数字字符的特殊情况,以下内容也有效,并且比我答案顶部的一般版本快一点。

使用format()方法在某种程度上涉及到转换回字符串的转换是为了处理零移动到字符串前面的情况。我把它主要表现为一种好奇心,因为除非你用数学方法掌握它的作用,否则它是相当难以理解的。它也没有处理负数。

n = int(s)
len_s = len(s)
ord_0 = ord('0')
di = ord(s[i])-ord_0
dj = ord(s[j])-ord_0
pi = 10**(len_s-(i+1))
pj = 10**(len_s-(j+1))
s = '{:0{width}d}'.format(n + (dj-di)*pi + (di-dj)*pj, width=len_s)

答案 1 :(得分:1)

它必须是某种类型的可变类型,我能想到的最好的是(虽然不能对性能提出任何要求):

def swapchar(text, fst, snd):
    ba = bytearray(text)
    ba[fst], ba[snd] = ba[snd], ba[fst]
    return ba

>>> swapchar('thequickbrownfox', 3, 7)
bytearray(b'thekuicqbrownfox')

您仍然可以将结果用作str / list - 或者如果需要,可以明确地将其转换为str

答案 2 :(得分:0)


    >>> int1 = 2
    >>> int2 = 3
    >>> eval(str(int1)+str(int2))
    23

答案 3 :(得分:0)

我知道你已经接受了一个答案,所以我不会在Python中编写代码,但是这里你可以用JavaScript创建它,它也有不可变的字符串:

function swapchar(string, j)
{
    return string.replace(RegExp("(.{" + j + "})(.)(.)"), "$1$3$2");
}

显然,如果j不在适当的范围内,那么它只返回原始字符串。