在利用Python解决各种实际问题的过程中,经常会遇到从某个对象中抽取部分值的情况,切片操作正是专门用于完成这一操作的有力武器。理论上而言,只要条件表达式得当,可以通过单次或多次切片操作实现任意切取目标值。切片操作的基本语法比较简单,但如果不彻底搞清楚内在逻辑,也极容易产生错误,而且这种错误有时隐蔽得比较深,难以察觉。本文通过详细例子总结归纳了切片操作的各种情况。若有错误和不足之处请大牛指正!


一、Python可切片对象的索引方式

Python可切片对象的索引方式包括:正索引负索引两部分。

如下图所示,以a = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]为例:


二、Python切片操作的一般方式

一个完整的切片表达式包含两个“:”,用于分隔三个参数(start_index、end_index、step),当只有一个“:”时,默认第三个参数step=1。

  1. 切片操作基本表达式:object[start_index : end_index : step]

step:正负数均可,其绝对值大小决定了切取数据时的“步长”,而正负号决定了“切取方向”,正表示“从左往右”取值,负表示“从右往左”取值。当step省略时,默认为1,即从左往右以增量1取值。“切取方向非常重要!”“切取方向非常重要!”“切取方向非常重要!”,重要的事情说三遍!

start_index:表示起始索引(包含该索引本身);该参数省略时,表示从对象“端点”开始取值,至于是从“起点”还是从“终点”开始,则由step参数的正负决定,step为正从“起点”开始,为负从“终点”开始。

end_index:表示终止索引(不包含该索引本身);该参数省略时,表示一直取到数据”端点“,至于是到”起点“还是到”终点“,同样由step参数的正负决定,step为正时直到”终点“,为负时直到”起点“。


三、Python切片操作详细例子

以下示例均以列表a = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]为例:

>>> a = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]

1.切取单个值
  1. >>> a[0]
  2. 0
  3. >>> a[-4]
  4. 6
2.切取完整对象
  1. >>> a[:] # 从左往右
  2. [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
  3. >>> a[::] # 从左往右
  4. [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
  5. >>> a[::-1] # 从右往左
  6. [9, 8, 7, 6, 5, 4, 3, 2, 1, 0]
3.start_index和end_index全为正(+)索引的情况
  1. >>> a[1:6] # step=1,从左往右取值,start_index=1到end_index=6同样表示从左往右取值。
  2. [1, 2, 3, 4, 5]
  3. >>>a[1:6:-1] # step=-1,决定了从右往左取值,而start_index=1到end_index=6决定了从左往右取值,两者矛盾。
  4. >>> [] # 输出为空列表,说明没取到数据。
  5. >>>a[6:1] # step=1,决定了从左往右取值,而start_index=6到end_index=1决定了从右往左取值,两者矛盾。
  6. >>> [] # 同样输出为空列表。
  7. >>>a[:6] # step=1,从左往右取值,从“起点”开始一直取到end_index=6。
  8. >>> [0, 1, 2, 3, 4, 5]
  9. >>>a[:6:-1] # step=-1,从右往左取值,从“终点”开始一直取到end_index=6。
  10. >>> [9, 8, 7]
  11. >>>a[6:] # step=1,从左往右取值,从start_index=6开始,一直取到“终点”。
  12. >>> [6, 7, 8, 9]
  13. >>>a[6::-1] # step=-1,从右往左取值,从start_index=6开始,一直取到“起点”。
  14. >>> [6, 5, 4, 3, 2, 1, 0]
4.start_index和end_index全为负(-)索引的情况
  1. >>>a[-1:-6] # step=1,从左往右取值,而start_index=-1到end_index=-6决定了从右往左取值,两者矛盾。
  2. >>> []
  3. >>>a[-1:-6:-1] # step=-1,从右往左取值,start_index=-1到end_index=-6同样是从右往左取值。
  4. >>> [9, 8, 7, 6, 5]
  5. >>>a[-6:-1] # step=1,从左往右取值,而start_index=-6到end_index=-1同样是从左往右取值。
  6. >>> [4, 5, 6, 7, 8]
  7. >>>a[:-6] # step=1,从左往右取值,从“起点”开始一直取到end_index=-6。
  8. >>> [0, 1, 2, 3]
  9. >>>a[:-6:-1] # step=-1,从右往左取值,从“终点”开始一直取到end_index=-6。
  10. >>> [9, 8, 7, 6, 5]
  11. >>>a[-6:] # step=1,从左往右取值,从start_index=-6开始,一直取到“终点”。
  12. >>> [4, 5, 6, 7, 8, 9]
  13. >>>a[-6::-1] # step=-1,从右往左取值,从start_index=-6开始,一直取到“起点”。
  14. >>> [4, 3, 2, 1, 0]
5.start_index和end_index正(+)负(-)混合索引的情况
  1. >>>a[1:-6] # start_index=1在end_index=-6的左边,因此从左往右取值,而step=1同样决定了从左往右取值。
  2. >>> [1, 2, 3]
  3. >>>a[1:-6:-1] # start_index=1在end_index=-6的左边,因此从左往右取值,但step=-则决定了从右往左取值,两者矛盾。
  4. >>> []
  5. >>>a[-1:6] # start_index=-1在end_index=6的右边,因此从右往左取值,但step=1则决定了从左往右取值,两者矛盾。
  6. >>> []
  7. >>>a[-1:6:-1] # start_index=-1在end_index=6的右边,因此从右往左取值,而step=-1同样决定了从右往左取值。
  8. >>> [9, 8, 7]
6.连续切片操作
  1. >>>a[:8][2:5][-1:]
  2. >>> [4]

相当于:

a[:8]=[0, 1, 2, 3, 4, 5, 6, 7]

a[:8][2:5]= [2, 3, 4]

a[:8][2:5][-1:] = 4

理论上可无限次连续切片操作,只要上一次返回的依然是非空可切片对象。

7.切片操作的三个参数可以用表达式
  1. >>>a[2+1:3*2:7%3] # 即:a[2+1:3*2:7%3] = a[3:6:1]
  2. >>> [3, 4, 5]
8.其他对象的切片操作

前面的切片操作说明都以list为例进行说明,但实际上可进行的切片操作的数据类型还有很多,包括元组、字符串等等。

  1. >>> (0, 1, 2, 3, 4, 5)[:3] # 元组的切片操作
  2. >>> (0, 1, 2)
  3. >>>'ABCDEFG'[::2] # 字符串的切片操作
  4. >>>'ACEG'
  5. >>>for i in range(1,100)[2::3][-10:]: # 利用range函数生成1-99的整数,然后取3的倍数,再取最后十个。
  6. print(i, end=' ')
  7. >>> 72 75 78 81 84 87 90 93 96 99

四、Python常用切片操作

以列表:a = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]为说明对象

1.取偶数位置
  1. >>>b = a[::2]
  2. [0, 2, 4, 6, 8]
2.取奇数位置
  1. >>>b = a[1::2]
  2. [1, 3, 5, 7, 9]
3.拷贝整个对象
  1. >>>b = a[:] # ★★★★★
  2. >>>print(b) # [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
  3. >>>print(id(a)) # 41946376
  4. >>>print(id(b)) # 41921864
  5. >>>b = a.copy()
  6. >>>print(b) # [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
  7. >>>print(id(a)) # 39783752
  8. >>>print(id(b)) # 39759176

需要注意的是:[:]和.copy()都属于“浅拷贝”,只拷贝最外层元素,内层嵌套元素则通过引用,而不是独立分配内存。

  1. >>>a = [1,2,['A','B']]
  2. >>>print('a={}'.format(a))
  3. a=[1, 2, ['A', 'B']] # 原始a
  4. >>>b = a[:]
  5. >>>b[0] = 9 # 修改b的最外层元素,将1变成9
  6. >>>b[2][0] = 'D' # 修改b的内嵌层元素
  7. >>>print('a={}'.format(a)) # b修改内部元素A为D后,a中的A也变成了D,说明共享内部嵌套元素,但外部元素1没变。
  8. a=[1, 2, ['D', 'B']]
  9. >>>print('b={}'.format(b)) # 修改后的b
  10. b=[9, 2, ['D', 'B']]
  11. >>>print('id(a)={}'.format(id(a)))
  12. id(a)=38669128
  13. >>>print('id(b)={}'.format(id(b)))
  14. id(b)=38669192
4.修改单个元素
  1. >>>a[3] = ['A','B']
  2. [0, 1, 2, ['A', 'B'], 4, 5, 6, 7, 8, 9]
5.在某个位置插入元素
  1. >>>a[3:3] = ['A','B','C']
  2. [0, 1, 2, 'A', 'B', 'C', 3, 4, 5, 6, 7, 8, 9]
  3. >>>a[0:0] = ['A','B']
  4. ['A', 'B', 0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
6.替换一部分元素
  1. >>>a[3:6] = ['A','B']
  2. [0, 1, 2, 'A', 'B', 6, 7, 8, 9]

五、总结

(一)start_index、end_index、step可同为正、同为负,也可正负混合使用。但必须遵循一个原则,即两者的取值顺序必须是相同的,否则无法正确切取到数据:当start_index的位置在end_index的左边时,表示从左往右取值,此时step必须是正数(同样表示从左往右);当start_index的位置在end_index的右边时,表示从右往左取值,此时step必须是负数(同样表示从右往左)。对于特殊情况,当start_index或end_index省略时,起始索引和终止索引由step的正负来决定,不会存在取值方向出现矛盾的情况,但正和负取到的结果是完全不同的,因为一个向左一个向右。

(二)在利用切片时,step的正负是必须要考虑的,尤其是当step省略时。比如a[-1:],很容易就误认为是从“终点”开始一直取到“起点”,即a[-1:]= [0, 1, 2, 3, 4, 5, 6, 7, 8, 9],但实际上a[-1:]=a[-1]=9,原因在于step=1表示从左往右取值,而起始索引start_index=-1本身就是对象的最右边的元素了,再往右已经没数据了,因此只有a[-1]一个元素。

 

 

 

原文经过些许修改

原文作者:马尔代夫Maldives

原文链接:https://www.jianshu.com/p/15715d6f4dad

彻底搞懂Python切片操作的更多相关文章

  1. 关于python切片操作笔记

    一. Python可切片对象的索引方式 包括:正索引和负索引两部分,如下图所示,以a = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]为例:   python索引方式.jpg 二. P ...

  2. 关于javascript里面仿python切片操作数组的使用方法

    其实在使用了好一段时间的 python之后,我觉得最让我念念不忘的并不是python每次在写函数或者循环的时候可以少用{}括号这样的东西(ps:其实也是了..感觉很清爽,而且又开始写js的时候老是想用 ...

  3. 一文搞懂Python迭代器和生成器

    很多童鞋搞不懂python迭代器和生成器到底是什么?它们之间又有什么样的关系? 这篇文章就是要用最简单的方式让你理解Python迭代器和生成器! 1.迭代器和迭代过程 维基百科解释道: 在Python ...

  4. python切片操作

    序列类型是其元素被顺序放置的一种数据结构类型,这种方式允许通过下标的方式来获得某一个数据元素,或者通过指定下标范围来获得一组序列的元素.这种访问序列的方式叫做切片.字符串也可以使用切片操作.切片操作符 ...

  5. 一篇搞懂python文件读写操作(r/r+/rb/w/w+/wb/a/a+/ab)

           关于文件操作的几种常用方式,网上已有很多解说,内容很丰富,但也因此有些杂乱复杂.今天,我就以我个人的学习经验写一篇详细又易懂的总结文章,希望大家看完之后会有所收获. 一.核心功能 ‘r’ ...

  6. 天啦噜!仅仅5张图,彻底搞懂Python中的深浅拷贝

    Python中的深浅拷贝 在讲深浅拷贝之前,我们先重温一下 is 和==的区别. 在判断对象是否相等比较的时候我们可以用is 和 == is:比较两个对象的引用是否相同,即 它们的id 是否一样 == ...

  7. 一文搞懂Python Unittest测试方法执行顺序

    大家好~我是米洛! 欢迎关注我的公众号测试开发坑货,一起交流!点赞收藏关注,不迷路. Unittest unittest大家应该都不陌生.它作为一款博主在5-6年前最常用的单元测试框架,现在正被pyt ...

  8. 一文搞懂 Python 的模块和包,在实战中的最佳实践

    最近公司有个项目,我需要写个小爬虫,将爬取到的数据进行统计分析.首先确定用 Python 写,其次不想用 Scrapy,因为要爬取的数据量和频率都不高,没必要上爬虫框架.于是,就自己搭了一个项目,通过 ...

  9. 一张图搞懂容器所有操作 - 每天5分钟玩转 Docker 容器技术(26)

    前面我们已经讨论了容器的各种操作,对容器的生命周期有了大致的理解,下面这张状态机很好地总结了容器各种状态之间是如何转换的. 如果掌握了前面的知识,要看懂这张图应该不难.不过有两点还是需要补充一下: 可 ...

随机推荐

  1. Redis---学习笔记(更新中)

    一.基本命令 #查看所有键 keys * #查看指定键 keys key #查看模糊键 keys ke* keys ke? keys ke[a-z] keys ke\? #判断键是否存在 exists ...

  2. 05-MySQL的完整性约束

    1.整体说明(1)讨论重点内容    not null 与default    unique:表中该值唯一,不能有重复值    primary    auto_increment    foreign ...

  3. ElasticSearch学习(一):ElasticSearch介绍

    一.ElasticSearch是什么? ElasticSearch是一款非常强大的.基于Lucene的开源搜索及分析引擎,可以帮助你从海量数据中,快速找到相关的数据信息. 比如,当你在GitHub上搜 ...

  4. Python 爬虫从入门到进阶之路(九)

    之前的文章我们介绍了一下 Python 中的正则表达式和与爬虫正则相关的 re 模块,本章我们就利用正则表达式和 re 模块来做一个案例,爬取<糗事百科>的糗事并存储到本地. 我们要爬取的 ...

  5. 【简易bat脚本】启动java程序

    前置条件:path中添加了JAVAHOME配置了java环境变量 1.新建txt文本文件 2.粘贴以下内容 @echo off set path=%path%;.;java -classpath &q ...

  6. 吐槽:那些Java设计中不得不说的槽点

    1. 求长度各有千秋 你是否曾经在面试的时候,经常被问到:数组有没有 length() 方法?字符串有没有 length() 方法? 集合有没有 length() 方法? 面对这个问题,那么不得不吐槽 ...

  7. BZOJ 1067:[SCOI2007]降雨量(RMQ+思维)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1067 题意:…… 思路:首先我们开一个数组记录年份,一个记录降雨量,因为年份是按升序排列的,所以我们 ...

  8. git报错---If no other git process is currently running...

    今天帮同事上传一个代码(预生产环境),当我执行到git add 文件 的时候,出现了如下错误: If no other git process is currently running, this p ...

  9. java中动态代理的使用

    代理模式是常用的java设计模式,他的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息.过滤消息.把消息转发给委托类,以及事后处理消息等.代理类与委托类之间通常会存在关联关系,一个代 ...

  10. 一个内存不能被written的问题

    C程序面试中曾经面试过这样一道题: #include <stdio.h> int main() { char *p = "12345"; *p = '6'; print ...