【问题标题】:Getting only element from a single-element list in Python?仅从 Python 中的单元素列表中获取元素?
【发布时间】:2016-01-14 16:09:53
【问题描述】:

当已知 Python 列表始终包含单个项目时,是否有其他方法可以访问它:

mylist[0]

您可能会问,“您为什么要这样做?”。唯有好奇心。似乎有另一种方法可以在 Python 中执行所有操作

【问题讨论】:

  • 可能有替代方案,但通常只有一种 明显 方法可以做到这一点 - 在这种情况下,您似乎已经找到了它。
  • 如果它总是包含单个项目,那么list 可能不是最好的数据类型?
  • @ekhumoro:我实际上偏向于序列解包方法,因为它验证了序列只有一个元素的假设。 mylist[0] 在您至少有一个元素时成功,但如果您实际上有 30 个元素,则不会抱怨。 singleitem, = mylist 验证您只有一个元素,不多也不少。
  • @ShadowRanger。问题是关于访问列表中已经已知包含单个项目的唯一元素。
  • @ekhumoro:没有异议。我只是更喜欢防御性地编程,因此违反要求的行为不会默默通过(肯定会失败,但会大声失败,这比细微的不当行为更容易识别和修复)。如果每次生产代码中的某些“已知”行为被证明是大错特错时,我都会得到一分钱......好吧,我可能不会富有,但我可以接受一家人出去享用真的美味的晚餐。

标签: python python-3.x python-2.7 list iterable-unpacking


【解决方案1】:

如果不是完全一项,则引发异常:

序列解包:

singleitem, = mylist
# Identical in behavior (byte code produced is the same),
# but arguably more readable since a lone trailing comma could be missed:
[singleitem] = mylist

所有其他人都默默地忽略规范违规,产生第一个或最后一个项目:

显式使用迭代器协议:

singleitem = next(iter(mylist))

破坏性弹出:

singleitem = mylist.pop()

负指数:

singleitem = mylist[-1]

通过单次迭代设置for(因为当循环终止时,循环变量以其最后一个值保持可用):

for singleitem in mylist: break

猖獗的精神错乱:

# But also the only way to retrieve a single item and raise an exception on failure
# for too many, not just too few, elements as an expression, rather than a statement,
# without resorting to defining/importing functions elsewhere to do the work
singleitem = (lambda x: x)(*mylist)

还有很多其他的(结合或改变上述部分,或以其他方式依赖隐式迭代),但你明白了。

【讨论】:

  • 第一个是独一无二的,如果列表包含多个元素,它也会引发。
  • 第一个也可以写成[singleitem] = mylist
  • @CarstenS:通常,我不赞成使用[] 进行序列解包(for [i, x] in enumerate(iterable): 让我发狂),但是是的,在解包单个值的情况下,尾随逗号很容易错过了,并且出于可读性原因使用括号是合理的。我已将其添加到答案中。谢谢!
  • @Bharel:好的,只要我们同意这是疯了,我们就很好。 :-)
  • @pabouk:技术上是(显然有人刚刚从这个评论链中删除了评论),这只是我想出的一个疯狂的解决方案(当我提到“猖獗的精神错乱”版本时,那就是一),这就是为什么我没有把它放在答案中。你只需这样做:(lambda x: x)(*mylist); lambdas 是作为表达式的函数定义,因此您可以定义一个,然后在一行中将解压缩的迭代传递给一个,它要么因异常而死(如果解包与参数计数不精确匹配)或调用函数成功并返回参数。
【解决方案2】:

我将添加more_itertools 库有一个工具可以从一个可迭代对象中返回一个项目。

from more_itertools import one


iterable = ["foo"]
one(iterable)
# "foo"

此外,more_itertools.one 会在可迭代对象为空或有多个项目时引发错误。

iterable = []
one(iterable)
# ValueError: not enough values to unpack (expected 1, got 0)

iterable = ["foo", "bar"]
one(iterable)
# ValueError: too many values to unpack (expected 1)

more_itertools是第三方包> pip install more-itertools

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2020-04-23
    • 1970-01-01
    • 2016-03-02
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2012-04-04
    相关资源
    最近更新 更多