284

编写以下代码的pythonic方式是什么?

extensions = ['.mp3','.avi']
file_name = 'test.mp3'

for extension in extensions:
    if file_name.endswith(extension):
        #do stuff

我有一个模糊的记忆,循环的显式声明for可以避免并写在if条件中。这是真的?

4

7 回答 7

568

虽然并不广为人知,但 str.endswith也接受一个元组。你不需要循环。

>>> 'test.mp3'.endswith(('.mp3', '.avi'))
True
于 2013-08-21T08:03:46.333 回答
58

只需使用:

if file_name.endswith(tuple(extensions)):
于 2013-08-21T08:03:36.980 回答
6

有两种方法:正则表达式和字符串(str)方法。

字符串方法通常更快( ~2x )。

import re, timeit
p = re.compile('.*(.mp3|.avi)$', re.IGNORECASE)
file_name = 'test.mp3'
print(bool(t.match(file_name))
%timeit bool(t.match(file_name)

每个循环 792 ns ± 1.83 ns(平均值 ± 标准偏差。7 次运行,每次 1000000 次循环)

file_name = 'test.mp3'
extensions = ('.mp3','.avi')
print(file_name.lower().endswith(extensions))
%timeit file_name.lower().endswith(extensions)

每个循环 274 ns ± 4.22 ns(平均值 ± 标准偏差。7 次运行,每次 1000000 次循环)

于 2018-03-03T21:52:04.200 回答
4

另一种可以返回匹配字符串列表的方法是

sample = "alexis has the control"
matched_strings = filter(sample.endswith, ["trol", "ol", "troll"])
print matched_strings
['trol', 'ol']
于 2018-12-31T09:08:46.813 回答
2

我只是在寻找其他东西时遇到了这个。

我建议使用os包中的方法。这是因为您可以使其更通用,以补偿任何奇怪的情况。

您可以执行以下操作:

import os

the_file = 'aaaa/bbbb/ccc.ddd'

extensions_list = ['ddd', 'eee', 'fff']

if os.path.splitext(the_file)[-1] in extensions_list:
    # Do your thing.
于 2018-03-13T09:00:26.497 回答
1

我有这个:

def has_extension(filename, extension):

    ext = "." + extension
    if filename.endswith(ext):
        return True
    else:
        return False
于 2017-04-07T08:34:41.293 回答
0

另一种可能性是使用 IN 语句:

extensions = ['.mp3','.avi']
file_name  = 'test.mp3'
if "." in file_name and file_name[file_name.rindex("."):] in extensions:
    print(True)
于 2018-04-23T13:09:28.690 回答