微信公众号: 冰咖啡与狗
一杯冰咖啡,一条狗,一个碎碎念的程序员...
不一样于 C++ 和 Java,在 Python 中,没有在标准库中内置数组类型,而是一般利用列表来表示数组,同时它也比数组的使用要灵活得太多。python
列表是 Python 中最基本的数据结构,列表中的元素的存储是有序的,因此咱们能够直接利用索引去访问列表中的值。在学习数据结构时,咱们知道,对一个数据结构的操做,无外乎增删改查四种。接下来本文将从这四个方面来介绍列表的一些经常使用小技巧。程序员
对列表的增长操做有两种,一种是建立一个新的列表,一种是在一个已有的列表中增长一个新元素。咱们首先介绍如何建立一个新的列表:编程
# 建立一个空列表 list1 = [] # 建立一个默认值为0,长度为n的列表 list2 = [0] * n # 利用列表生成式建立列表 list3 = [i for i in range(10)] # 5*5 二维列表 list4 = [[i for i in range(5)] for j in range(5)] # 利用其余数据结构生成列表, list() 函数接收一个可迭代对象做为参数 tuple1 = (1, 2, 3) list5 = list(tuple1) # 字符串转化为列表 str1 = "I love code" # 每一个元素做为列表中的一个元素 list6 = list(str1) # 按照指定字符分割 list7 = str1.split(" ") # ['I', 'love', 'code']
在编程过程当中,有时咱们须要对一个列表进行操做,可是列表是一个可变对象,对列表的操做每每会改变其原本的顺序结构。所以,当咱们不想改变列表原有的顺序结构,咱们须要对旧的列表进行一个拷贝,而后在新的列表上进行操做。拷贝存在两种,浅拷贝和深拷贝。可能有人会对这两种拷贝存在疑问,他们获得的列表的形式不是同样的嘛?数组
是的,它们在拷贝后获得的列表形式确实是同样的。可是在进行操做时,就会有很大的差异了,特别是当你的列表中存在可变对象时。咱们只须要记住一点,浅拷贝只复制不可变对象,而深拷贝不只复制不可变对象,还复制了可变对象。如下进行举例说明:微信
# 若是一个列表中的元素都是不可变对象,能够直接用浅拷贝 list1 = [1, 2, 'i'] list2 = list1[::1] # 浅拷贝 list3 = [i for i in list1] # 浅拷贝 import copy list4 = copy.copy(list1) # 浅拷贝 # 若是一个列表中的元素包含可变对象,要完成复制列表,须要用深拷贝 list5 = ["Will", 1, ["Python", "Java", "C++"]] list6 = copy.deepcopy(list5)
为何在包含不可变对象的时候,须要用到深拷贝呢?数据结构
上面咱们说过,浅拷贝只拷贝不可变对象。当列表中存在可变对象时,咱们能够发现其引用 id 是同样的,即它们是同一个对象,所以对 list5 的操做会影响到 list6 。而在深拷贝中,对于可变对象,在拷贝的新列表中会生成一个新的对象,因此对 list6 的修改不会影响到 list5 。app
id(list5[2]) # 2195936916360 id(list6[2]) # 2195936916744
对列表的另外一种增长操做就是在已有的列表的基础上增长元素了,主要包括如下几种操做:函数
list1 = ["I"] # 在列表尾部增长一个新元素 list1.append("love") # 合并两个列表 list2 = ["Apple", 3] list1.extend(list2) # ["I", "love", "Apple", 3] list1 = list1 + list2 # ["I", "love", "Apple", 3] # 在指定位置插入元素 list1.insert(index=1, "not") # ["I", "not", "love", "Apple", 3]
对数据结构的删除操做包括两种,删除整个列表,或者删除列表中的某些元素。对于从内存中删除整个列表,该操做十分简单:学习
list1 = [1, 2, 3] del list1
对于删除列表中的元素,一样可使用 del
语句,同时还可使用 remove()
函数。spa
list1 = [1, 2, 3] print(list1) # [1, 2, 3] del list1[1] print(list1) # [1, 3] # 移除某个值的第一匹配项 list2 = [1, 2, 2, 3, 5] list2.remove(2) # 移除第一个值为 2 的元素
修改操做即在原有的列表的基础上对其进行修改,上面以及介绍了增长和删除操做,这里就对不改变元素的个数的操做进行介绍,主要包括排序和反转列表等。
# 对一个整数列表进行排序 list1 = [9, 5, 6, 2, 7, 1, 3] # 升序排列 list1.sort() # [1, 2, 3, 5, 6, 7, 9] list1 = sorted(list1) # [1, 2, 3, 5, 6, 7, 9] # 降序排列 list1.sort(reverse=True) # [9, 7, 6, 5, 3, 2, 1] list1 = sorted(list1, reverse=True) # [9, 7, 6, 5, 3, 2, 1] # 指定键值排序 list1 = [('a', 3), ('b', 2), ('c', 1)] list1.sort(key=lambda x:x[1]) # [('c', 1), ('b', 2), ('a', 3)] ## 反转列表,即将原来的列表逆序排列 list2 = [4, 2, 3, 1] list2 = list2[::-1] # [1, 3, 2, 4] list2.reverse() # [1, 3, 2, 4]
对列表的查找操做无外乎如下几种:
列表的遍历方式有三种,分别是根据索引遍历、根据值遍历和 enumerate()
遍历。
list1 = ["Python", "C++", "Java"] # 方法 1 for i in range(len(list1)): print(list1[i]) # 方法 2 , 遍历速度最高效,可是没法记录索引 for i in list1: print(i) # 方法 3 for index, val in enumerate(list1): print(val)
查找元素有两种,一种是按照索引查找,一种是遍历按值查找。
# 查找指定索引的元素 list1 = [1, 3, 2, 5, 7, 8] list1[3] # 返回索引为 3 的元素,值为5 # 查找指定值的元素 key = 5 for i in list1: if i == key: print("OK") # 查找最大元素和最小元素 max(list1) # 8 min(list1) # 1
上述介绍了列表的增删改查四种经常使用操做,固然对列表的操做不单单只有这些,例如还有两个列表的比较,列表结构的转化等等。
比较两个列表是不是同样的,这里的前提条件是这两个列表对象不是同一个对象。
list5 = [("car", 1), ("bike", 2), ("foot", 3)] list6 = [("car", 1), ("bike", 2), ("foot", 3)] # 判断两个列表是不是同一个对象 list5 is list6 # False # 判断两个列表是否相等 list5 == list6 # True # 当列表中的元素不少,且结构复杂时,可使用 numpy 模块 import numpy as np a = np.array(list5) b = np.array(list6) (a == b).all() # True
列表结构的转化主要包括将列表转化为元组、集合、字符串和字典:
list1 = ["a", "b", "c", "d", "c"] # 列表转化为元组 tuple1 = tuple(list1) # ('a', 'b', 'c', 'd', 'c') # 列表转化为集合 set1 = set(list1) # {'a', 'b', 'c', 'd'} # 列表转化为字符串 str1 = "".join(list1) # 'abcdc' # 列表转化为字典,元素为 key, 次数为 value from collections import Counter, defaultdict dict1 = Counter(list1) # Counter({'a': 1, 'b': 1, 'c': 2, 'd': 1}) # 或者 dict2 = defaultdict(int) for i in list1: dict2[i] += 1 dict2 # defaultdict(int, {'a': 1, 'b': 1, 'c': 2, 'd': 1})