百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

Python: 列表、数组及迭代器切片的区别及联系

off999 2024-11-06 11:22 29 浏览 0 评论

1. 对列表和数组进行切片

1.1 切片索引

众所周知,Python中的列表和numpy数组都支持用begin: end语法来表示[begin, end)区间的的切片索引:

import numpy as np
my_list= [1, 2, 3, 4, 5]
print(my_list[2: 4]) # [3, 4]

my_arr = np.array([1, 2, 3, 4, 5])
print(my_arr[2: 4]) # [3 4]

以上操作实际上等同于用slice切片索引对象对其进行切片:

print(my_list[slice(2, 4)]) # [3, 4]
print(my_arr[slice(2, 4)]) # [3 4]

numpy数组还支持用列表和numpy数组来表示切片索引,而列表则不支持:

print(my_arr[[2, 3]]) # [3 4]
print(my_arr[np.arange(2, 4)]) # [3, 4]

print(my_list[[2, 3]]) # TypeError: list indices must be integers or slices, not list
print(my_list[np.arange(2, 4)]) # TypeError: only integer scalar arrays can be converted to a scalar index

Pytorch的torch.utils.data.Dataset数据集支持单元素索引,但不支持切片:

from torchvision.datasets import FashionMNIST
from torchvision.transforms import Compose, ToTensor, Normalize

transform = Compose(
        [ToTensor(),
         Normalize((0.1307,), (0.3081,))
         ]
)
        
data = FashionMNIST(
        root="data",
        download=True,
        train=True,
        transform=transform
    )

print(data[0], data[1]) # (tensor(...), 0) (tensor(...), 0)
print(data[[0, 1]]) # ValueError: only one element tensors can be converted to Python scalars
print(data[: 2]) # ValueError: only one element tensors can be converted to Python scalars

要想对torch.utils.data.Dataset进行切片,需要创建Subset对象:

import torch
indices = [0, 1] # or indices = np.arange(2)
data_0to1 = torch.utils.data.Subset(data, indices)
print(type(data_0to1)) # <class 'torch.utils.data.dataset.Subset'>

Subset对象同样支持单元素索引操作且不支持切片:

print(data_0to1[0]) # (tensor(...), 0)

查看Pytorch源码可知,Subset类的定义实际上是这样的:

class Subset(Dataset[T_co]):
    r"""
    Subset of a dataset at specified indices.

    Args:
        dataset (Dataset): The whole Dataset
        indices (sequence): Indices in the whole set selected for subset
    """
    dataset: Dataset[T_co]
    indices: Sequence[int]

    def __init__(self, dataset: Dataset[T_co], indices: Sequence[int]) -> None:
        self.dataset = dataset
        self.indices = indices

    def __getitem__(self, idx):
        return self.dataset[self.indices[idx]]

    def __len__(self):
        return len(self.indices)

从以上代码片段可以清晰地看到Subset类用indices来存储本身做为子集的索引集合,然后重写(override)了__getitem__()方法来实现对子集的单元素索引。

1.2 对切片索引进行命名

有时我们会使用充满硬编码的切片索引,这使得代码难以阅读,比如下面这段代码:

record = ".....100...513.25.."
cost = int(record[5: 8]) * float(record[11: 17])
print(cost)  # 51325.0

与其这样做,我们不如对切片进行命名:

SHARES = slice(5, 8)
PRICE = slice(11, 17)
cost = int(record[SHARES]) * float(record[PRICE])
print(cost) # 51325.0

在后一种版本中,由于避免了使用许多神秘难懂的硬编码索引,我们的代码就变得清晰了许多。

正如我们前面所说,这里的slice()函数会创建一个slice类型的切片对象,可以用在任何运行切片的地方:

items = [0, 1, 2, 3, 4, 5, 6]
a = slice(2, 4)
print(items[2: 4]) # [2, 3]
print(items[a]) # [2, 3]
items[a] = [10, 11] 
print(items) # [0, 1, 10, 11, 4, 5, 6]
del items[a]
print(items) # [0, 1, 4, 5, 6]

如果有一个slice对象的实例s,可以分别用过s.starts.stop以及s.step属性来跌倒关于该对象的信息。例如:

a = slice(5, 50, 2)
print(a.start, a.stop, a.step) # 5 10 2

此外,可以通过使用indices(size)方法将切片映射到特定大小的序列上。这会返回一个[start, stop, step)元组,所有的值都已经恰当地限制在边界以内(当做索引操作时可避免出现IndexError异常)。例如:

s = 'HelloWorld'
print(a.indices(len(s)))
print(*a.indices(len(s)))
for i in range(*a.indices(len(s))):
    print(s[i])
# W
# r
# d

2. 对迭代器做切片操作

要对迭代器和生成器做切片操作,普通的切片操作符在这里是不管用的:

def count(n):
    while True:
        yield n
        n += 1
c = count(0)
print(c[10: 20]) # TypeError: 'generator' object is not subscriptable

此时,itertools.islice()函数是最完美的选择:

import itertools
for x in itertools.islice(c, 10, 20):
    print(x)
# 10
# 11
# 12
# 13
# 14
# 15
# 16
# 17
# 18
# 19

注意,迭代器和生成器之所以没法执行普通的切片操作,这是因为不知道它们的长度是多少(而且它们也没有实现索引)。islice()产生的结果是一个迭代器,它可以产生出所需要的切片元素,但这是通过访问并丢弃起始索引之前的元素来实现的。之后的元素会由islice对象产生出来,直到到达结束索引为止。

还有一点需要重点强调的是islice()会消耗掉所提供的的迭代器中数据。由于迭代器中的元素只能访问一次,没法倒回去,因此这里就需要引起我们的注意了。如果之后还需要倒回去访问前面的元素,那也许就应该先将数据转到列表中去。

文章来自https://www.cnblogs.com/orion-orion/p/16464225.html

相关推荐

电视剧免费观看电视剧在线(电视剧免费观看电视剧在线的软件)
  • 电视剧免费观看电视剧在线(电视剧免费观看电视剧在线的软件)
  • 电视剧免费观看电视剧在线(电视剧免费观看电视剧在线的软件)
  • 电视剧免费观看电视剧在线(电视剧免费观看电视剧在线的软件)
  • 电视剧免费观看电视剧在线(电视剧免费观看电视剧在线的软件)
改名字大全生辰八字免费(改名字大全生辰八字免费五行属火的字有哪些)

起名步骤:1、分析生辰八字命主五行、格局,找出用神喜忌。2、选出命局所喜用所需要的五行汉字。3、根据姓氏及辈分、时代特征、性别、八字数理合理搭配将要用的字,去其忌讳用字,避开凶数.取其字的意、形、象、...

主题字体免费(小猫の字体)

在搜索栏里面输入免费字体和主题就出来了1.阿里巴巴普惠体在其官方下载渠道上,明确声明:允许任何个人和企业免费使用,包括商用用途,但禁止用于违法用途。2.方正免费字体免费字体:方正黑体、方正书宋、方正仿...

迅雷输入法下载(迅雷输入法最新版本)

您好,不管是哪个输入法,都会收集用户的密码和个人资料以及隐私内容的,希望能帮到您,祝您生活愉快!迅雷输入是指在计算机领域中,将数据从一个地方快速、高效地传输到另一个地方的过程,通常用于下载文件、上传...

丝瓜app破解版无限次(丝瓜app破解版apk)

1、单个软件缓存数据过多。可进入设置--应用程序--全部--查找点击出现闪退的软件名称--清除数据后再使用试试。2、安装位置不合适(部分机型才有该功能))部分软件对手机安装位置有所要求,安装软件更好是...

免费英语学习网站(有什么免费学英语的网站)
免费英语学习网站(有什么免费学英语的网站)

现在网上的高中学习网站很多,像高中教育网校,简单学习网,新东方在线,学而思网校,中小学教育网等。各有利弊,选择适合自己的。并且能不能提高成绩,关键还是靠学生自己努力。我们来看下这些课堂总体特点:封闭课堂在听课过程中屏蔽了聊天、游戏等对学生听...

2026-01-18 09:03 off999

在线铃声下载 免费下载(手机铃声制作app下载)
在线铃声下载 免费下载(手机铃声制作app下载)

移动那叫“彩铃”,联通那叫“炫铃”,电信那叫“悦铃(可能不对)”总之,这些都是由运营商提供的,有的是免费有的是付费,需要订购。也就是需要你手机发指令或到网上营业厅或直接到柜台办理,然后在设置里设置指定某电话听到的回铃音是什么音乐(这个设置是...

2026-01-18 08:51 off999

下载cad制图软件(下载cad制图软件要钱吗)
  • 下载cad制图软件(下载cad制图软件要钱吗)
  • 下载cad制图软件(下载cad制图软件要钱吗)
  • 下载cad制图软件(下载cad制图软件要钱吗)
  • 下载cad制图软件(下载cad制图软件要钱吗)
种子搜索神器在线搜索(种子搜索神器在线搜索下载)

01.磁力熊磁力熊,是一个内容丰富、功能最为强大的一个磁力搜索网站,通过它不仅仅可以搜索到大量纯净的1080P高分电影,像一些比较小众的影视剧这里也都能找到。02.夕阳小站夕阳小站,虽然网站整体界面设...

模拟炒股软件app排名(模拟炒股软件app排名榜)
模拟炒股软件app排名(模拟炒股软件app排名榜)

同花顺、云掌财经、东方财富网、大智慧同花顺是股民使用比较多的软件了,实时的行情更新,数据分析,模拟炒股都是同花顺的有点所在,在炒股软件使用中占有率非常高。自定义设置的8个板块等按快捷代码查看相当方便(一键查看自选股)。炒股入门首选。大智慧也...

2026-01-18 08:03 off999

oa办公系统功能介绍(oa办公系统都有哪些)

以下是一些可能包含在OA办公系统中的常见功能:1.通讯录管理:包括添加、编辑和删除联系人信息,可按部门、职位等分类浏览联系人。2.日程管理:提供个人和团队日程安排功能,可创建、编辑和共享日程,设置...

下载word(收入证明模板免费下载word)
  • 下载word(收入证明模板免费下载word)
  • 下载word(收入证明模板免费下载word)
  • 下载word(收入证明模板免费下载word)
  • 下载word(收入证明模板免费下载word)
绝地求生(国际服)下载官方(绝地求生国际服下载安装免费正版)

首先需要安装和登录Steam(一个游戏平台);然后在Steam搜索"playerunknown'sbattlegrounds"并选择购买;下载并安装游戏,选择要加入的Serv...

麻花影视下载(麻花影视下载官方破解版)

被人举报了,然后关掉了国内的服务器,现在国内用的都是海外服务器而且用的人太多了所以卡

诺基亚n72(诺基亚n72上市时间价格多少)

n72是N系列中唯一一款不支持3G的智能机,还有N70。另外说说N72的十大缺点:1、电池待机时间较短,键盘较小,按键不方便2、嘈杂状态下铃声及振动较小,通话声音也较小3、短信书写中没有常用的网络符号...

取消回复欢迎 发表评论: