我有许多元素的列表。
我关心它的两个元素,a
和b
.
我不知道列表的顺序,也不想排序。
是否有一个不错的单行线,True
如果a
发生在之前会返回b
,否则会返回错误?
我有许多元素的列表。
我关心它的两个元素,a
和b
.
我不知道列表的顺序,也不想排序。
是否有一个不错的单行线,True
如果a
发生在之前会返回b
,否则会返回错误?
为了多样性,您还可以:
b in l[l.index(a):]
这将是True
如果a == b
。如果你知道a != b
,
b in l[l.index(a) + 1:]
好的,所以这个问题需要更多的工作。Mark Byers 完全正确,因为我的第一个测试只涵盖了结果为True
. 这一点尤其重要,因为我们需要其他解决方案的异常处理程序。所以我已经更详细地介绍了:
stmts = {
"Mark Byers: ": "x = l.index(a) < l.index(b)",
"jcollado: ": """try:
x = bool(l.index(b, l.index(a)))
except ValueError:
x = False""",
"Greg Hewgill: ": """try:
x = b in l[l.index(a):]
except ValueError:
x = False"""
}
setups = ["a = 80; b = 90; l = list(range(100))",
"a = 5; b = 10; l = list(range(100))",
"a = 90; b = 80; l = list(range(100))",
"a = 10; b = 5; l = list(range(100))"]
import timeit
for se in setups:
print(se)
for st in stmts:
print(st, timeit.timeit(stmt=stmts[st], setup=se))
print()
结果是:
a = 80; b = 90; l = list(range(100))
Mark Byers: 5.760545506106019
Greg Hewgill: 3.454101240451526 # Tie!
jcollado: 3.4574156981854536 # Tie!
a = 5; b = 10; l = list(range(100))
Mark Byers: 1.0853995762934794 # Close runner-up!
Greg Hewgill: 1.7265326426395209
jcollado: 1.0528704983320782 # Winner!
a = 90; b = 80; l = list(range(100))
Mark Byers: 5.741535600372806
Greg Hewgill: 3.623253643486848 # Winner!
jcollado: 4.567104188774817
a = 10; b = 5; l = list(range(100))
Mark Byers: 1.0592141197866987 # Winner!
Greg Hewgill: 4.73399648151641
jcollado: 4.77415749512712
因此,jcollado 方法的效率收益主要被异常处理程序的成本所消耗(特别是如果它触发)。所有三种解决方案都有一半的时间获胜(或与获胜者并列),因此很难说哪种方法最适合您的实际数据。也许您可能想选择最容易阅读的那个。
您可以使用list.index
:
l.index(a) < l.index(b)
这当然假设这两个项目都存在于列表中。
Mark Byers 的响应效果很好,但如果列表很长并且两个元素都接近结尾,则效率不会很高。
要仅遍历列表一次,您可以使用以下命令:
l.index(b, l.index(a))
根据需要,这是一个单行程序,但ValueError
无论如何您都需要捕获异常。