使用Python替换列表中的值

ak.*_*ak. 122 python performance replace list

我有一个列表,我想用None替换值,其中condition()返回True.

[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
Run Code Online (Sandbox Code Playgroud)

例如,如果条件检查bool(项目%2)应该返回:

[None, 1, None, 3, None, 5, None, 7, None, 9, None]
Run Code Online (Sandbox Code Playgroud)

最有效的方法是什么?

Joh*_*kin 179

使用列表解析构建新列表:

new_items = [x if x % 2 else None for x in items]
Run Code Online (Sandbox Code Playgroud)

如果需要,您可以就地修改原始列表,但实际上并不节省时间:

items = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
for index, item in enumerate(items):
    if not (item % 2):
        items[index] = None
Run Code Online (Sandbox Code Playgroud)

以下是(Python 3.6.3)演示非时间的时间:

In [1]: %%timeit
   ...: items = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
   ...: for index, item in enumerate(items):
   ...:     if not (item % 2):
   ...:         items[index] = None
   ...:
1.06 µs ± 33.7 ns per loop (mean ± std. dev. of 7 runs, 1000000 loops each)

In [2]: %%timeit
   ...: items = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
   ...: new_items = [x if x % 2 else None for x in items]
   ...:
891 ns ± 13.6 ns per loop (mean ± std. dev. of 7 runs, 1000000 loops each)
Run Code Online (Sandbox Code Playgroud)

和Python 2.7.6时序:

In [1]: %%timeit
   ...: items = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
   ...: for index, item in enumerate(items):
   ...:     if not (item % 2):
   ...:         items[index] = None
   ...: 
1000000 loops, best of 3: 1.27 µs per loop
In [2]: %%timeit
   ...: items = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
   ...: new_items = [x if x % 2 else None for x in items]
   ...: 
1000000 loops, best of 3: 1.14 µs per loop
Run Code Online (Sandbox Code Playgroud)

  • 我认为你有"最容易阅读"和"最有效"的反向 (3认同)
  • 那是最有效的吗?不枚举必须创建一个迭代器并形成一个元组,增加开销?是python arraylists中的列表,给你恒定的时间访问? (2认同)
  • @ geowa4:Python"列表"实际上是数组.`enumerate()`会产生很小的开销,但如果这是不可接受的,则可以手动跟踪索引.@ak:我不明白这个问题.`imap()`不是就地操作. (2认同)
  • 谢谢你的回答.但是这里有没有足够的用例来让python列表有一个替换方法?(类似于str.replace).避免需要的东西:`def replace(items,a,b):return [b if x == a else x for x in items] (2认同)

Lau*_*ves 65

ls = [x if (condition) else None for x in ls]
Run Code Online (Sandbox Code Playgroud)


bal*_*pha 12

这是另一种方式:

>>> L = range (11)
>>> map(lambda x: x if x%2 else None, L)
[None, 1, None, 3, None, 5, None, 7, None, 9, None]
Run Code Online (Sandbox Code Playgroud)

  • @gath:不要渴望为各种目的编写单行内容.有时,它们会提高可读性或性能,但通常不会.至于提示:了解Python的工具,尤其是列表(以及Python 3也有用)的理解,三元运算符,匿名(lambda)函数以及map,zip,filter,reduce等函数. (7认同)
  • +1,伙计们如何学习这一行漂亮的python代码...提示 (2认同)

Ale*_*lli 10

关于OP在评论中提出的问题,即:

如果我有一个生成器从范围(11)而不是列表中生成值,该怎么办?是否可以替换发电机中的值?

当然,这很简单......:

def replaceiniter(it, predicate, replacement=None):
  for item in it:
    if predicate(item): yield replacement
    else: yield item
Run Code Online (Sandbox Code Playgroud)

只需传递任何可迭代(包括调用生成器的结果)作为第一个arg,谓词决定是否必须将值替换为第二个arg,然后让我们翻录.

例如:

>>> list(replaceiniter(xrange(11), lambda x: x%2))
[0, None, 2, None, 4, None, 6, None, 8, None, 10]
Run Code Online (Sandbox Code Playgroud)