关于python:切片中的默认值是什么?

What are the default values set to in a slice?

我看了一下关于why-do-list-1-not-equal-listlenlist-1和python中的默认切片索引的顶级答案,了解切片中默认值的设置。两个最重要的答案都参考以下文档:

考虑到s[i:j:k]

If i or j are omitted or None, they become"end" values (which end depends on the sign of k). Note, k cannot be zero. If k is None, it is treated like 1.

假设我有以下代码:

1
2
3
4
5
6
s ="hello"
forward_s = s[::1]
print(forward_s) # =>"hello"

backward_s = s[::-1]
print(backward_s) # =>"olleh"

我知道,如果省略了索引,那么python将其视为在这些地方使用了None的值。根据文件,[i:j:k]中的ij的指数根据k的符号设置为"结束"值。对于正的k,我假设i设置为0j设置为字符串长度。但是,为负的k设置的值是什么?

例如,以下内容也会反转字符串:

1
reversed_str = s[-1:-6:-1]

那么,当k = -1时,i的默认值可能设置为-1j的默认值设置为-len(s) - 1

  • 这似乎是正确的分析。否则它将尝试从0到-1到-2…因此,[::-1]不会正确地反转列表,而是给出第一个元素,然后反转其余元素。
  • 什么是"最终价值"?
  • 当k为负时,i和j被视为字符串终止端的结束值。在你链接的答案中引用的文档是明确的。
  • 好吧,如果len(str) - 10是"最终值",那么为什么这不起作用:"hello"[len("hello") - 1: 0: -1]?这将返回"olle"而不是"olleh"。
  • 我想澄清一下。-1-len(s) - 1是否是另一组结束值,特别是k的负值。


默认值为NoneNoneNone

1
2
3
4
5
6
class Sliceable(object):
    def __getitem__(self, slice):
        print(slice)

Sliceable()[::]
>>> slice(None, None, None)

这与事实无关,即slice()确实需要stop论证。库文档对此不太明确,但C API明确指出,所有三个值都可能是空的:

The start, stop, and step parameters are used as the values of the slice object attributes of the same names. Any of the values may be NULL, in which case the None will be used for the corresponding attribute.

由可切片的对象来理解默认值。约定是使用第一个元素、最后一个元素、内部集合实现的最小步进:

1
2
3
4
5
l = [1, 2, 3]
l[slice(None, None, None])
>>> [1, 2, 3]
s[None:None:None])
>>> [1, 2, 3]

负步进将导致默认的startend值在语义上颠倒,即:

1
2
3
4
5
s = 'abc'
s[slice(None, None, -1)]
>>> 'cba'
s[::-1]
>>> 'cba'

注意,这并不意味着一个简单的值翻转,end的默认值通常是"在任何方向上超过序列末尾的一个",因为range()不包含结束值,但切片的默认值应包括完整序列。

记录如下:

s[i:j:k]

The slice of s from i to j with step k is defined as the sequence of items with index x = i + n*k such that 0 <= n < (j-i)/k. In other words, the indices are i, i+k, i+2*k, i+3*k and so on, stopping when j is reached (but never including j). When k is positive, i and j are reduced to len(s) if they are greater. When k is negative, i and j are reduced to len(s) - 1 if they are greater. If i or j are omitted or None, they become"end" values (which end depends on the sign of k). Note, k cannot be zero. If k is None, it is treated like 1.

  • 这里的None是什么意思:"foo"[None:None:-1]?所以我假设的第一个"无"是指字符串中的最后一个字符。第二个"无"翻译成什么?
  • 第二个None是否解释为-len(s) - 1s是有问题的字符串。
  • @asadmoosvi如前所述:对于内置于c的集合(以及任何模仿该集合的对象),None作为end参数被解释为len(s),传递给__getitem__()的实际值仍然是None。请注意,切片范围是端排它的。
  • 但这没有道理。如果end参数是-1,那么start参数是什么?换绳子?
  • @不好意思,打字错误,是len(s)(-1是len(s) - 1)。startNone,解释为0
  • step为负数时,startend的显式值是什么?专门针对字符串
  • 显式值为None。你在问隐含的价值观吗?
  • end是步骤参数正值的len(s)。我明白。但消极的一步是什么呢?这就是我困惑的地方。
  • @当出现负步进时,startend会翻转。具体细节如下。
  • 这也没有意义,因为如果它们翻转,那么"hello"[len("hello"):0:-1]应该与"hello"[::-1]相同。这里,startend已经"翻转",但结果不同。
  • @是的。读作"对于结束值,无论从哪个方向,总是有一个通过序列的末尾"
  • 我的意思是我能直觉地理解它的作用。好像我知道这里发生了什么。但我想确切地知道python在使用end参数作为负步骤的默认值。我只是讨厌接受一些东西来学习,因为这就是它"只是工作"的方式。
  • @asadmoosvi,为什么不直接进入实现并检查呢?因为对您的初始问题的解释字面上停在"由可切片的对象来理解默认值。"。
  • 如果我知道C的话我会的。
  • @一句话,语义解释(加在答案上)就不好了;-)