Python自定义序列类型

1. 序列类型

1.1 序列之间有什么区别?

序列的类型表:

序列类别序列类型
容器序列list ,tuple,deque
扁平序列str,bytes,bytearray,array.array
可变序列list,deque,bytearray,array
不可变序列str,tuple,bytes

这里简单的解释一下各个序列的作用:

1.容器序列

🌙可以存放不同的数据类型和不同的对象引用,举个代码例子:

class Demo:
    pass

class Demo01:
    pass

if __name__ == '__main__':
   a = [Demo01, Demo, 5, "name", 0.01]
   print(type(a)) #list类型
   b = (Demo01, Demo, 5, "name", 0.01)
   print(type(b)) #tuple类型
   c=collections.deque([Demo01, Demo, 5, "name", 0.01]) 
   print(type(c)) #deque类型
   print(a)  #测试输出结果 都能打印出对应的值
   print(b)
   print(c)

2.扁平序列

🌙该序列只能容纳一种类型,存放的只有值类型,下表中的 str(字符串)不过多解释,bytes 函数返回一个新的 bytes 对象,该对象是一个 0 <= x < 256 区间内的整数不可变序列。它是 bytearray 的不可变版本,bytearray字节数组也是可变的字节序列,这里主要解释array.array关于array.array的初始化类型:

# 只举几个例子
arr1 = array.array("I") #表示存放int类型
arr1.append(5)
arr1.append(8)
print(arr1)
print(type(arr1))
arr2 = array.array("b") #表示存放int类型
print(arr2)
print(type(arr2))
arr3 = array.array("B") #表示存放int类型
arr3.append(10)
arr3.append(5)
print(arr3)
print(type(arr3))
arr4 = array.array("f") #表示存放float类型
arr4.append(0.5)
arr4.append(1)
print(arr4) 
print(type(arr4))  #类型都是array.array 通过传递的字符串参数确定要使用的存储类型

🌙上述代码中( I b B )都是int类型,其实在它们的底层是用C语言存放的不同类型。

3.可变序列

🌙可以进行增删改地址不会变化这里举list和array的例子:

a = [1, 2, 3, 4, 5, 6]
b = array.array('I', [6, 7, 8, 9, 10])

if __name__ == '__main__':
    print("list初始地址", id(a))
    print("array初始地址", id(b))
    a.append(10)
    b.append(6)
print(a)
print("list添加值后的地址", id(a))
print(b)
print("array添加值后的地址", id(b))

输出结果:

在这里插入图片描述

4.不可变序列

🌙只能重新申请新的空间地址,无法增删改初始的值
str,tuple,bytes这些类型初始定义就不能进行直接的修改或者删除某个值,所有被定义为不可变序列

2. 序列+、+=、extend的区别

2.1 +和+=的区别

a = [1, 2]
c = a + [3, 4]  
print(id(c))
print(id(a))
print(c) #重新创建了一个list

# += 就地加
print(id(a))
a += [3, 4]
print(a)
print(id(a)) #同一个地址

输出结果:
在这里插入图片描述

第二个重点区别:

from collections import abc

a = [1, 2]
a += (3, 4)  #把[] 改成()  
print(a) #执行成功

c = a + (3, 4)  #把[] 改成()
#报错 TypeError: can only concatenate list (not "tuple") to list

🌙“+=” 能成功添加的原因是因为MutableSequence中的iadd魔法函数调用了extend这个方法,extend可以添加任意可迭代的数据。
看一下extend方法的使用:

a = [1, 2]

if __name__ == '__main__':
    a.extend((1, 2, 3)) #添加tuple
    a.extend([5, 6, 7]) #添加 list
    a.extend(({"Name": "张三"})) #添加dict
    a.extend(range(5, 10)) #range添加
    b = a.extend((7, 7, 7)) #注意此时的b是一个返回值类型
    print(b) #None extend不存在放回值
    print(a)

2.2 extend和append方法的区别

a = [1, 2]

if __name__ == '__main__':
    a.extend((1, 2, 3))
    print(a) #输出[1, 2, 1, 2, 3]
    a.append((5, 6, 7))
    print(a) #输出[1, 2, 1, 2, 3, (5, 6, 7)] 

🌙 extend内部会循环添加的类型,append直接添加不会遍历插入

3. 可切片对象

alist = [1, 2, 3, 4, 5, 6]

if __name__ == '__main__':
# 第一个: 表示起始默认0  第二个: 表示截取长度默认len  第3个表示截取步长默认1
print(alist[::])  # 截取从0开始 len长度 步长为1 所有直接输出原列表
print(alist[::-1])  # 返回原列表的逆序排列 步长为负数 从后往前
print(alist[::2])  # 返回原列表的偶数位数据 (2为步长) 0为起始位
print(alist[1::2])  # 获取奇数位置的数据 0为起始位
print(alist[3:6])  # 从3开始截取长度为6
print(alist[0:100])  # 切片位置大于列表长度时,从列表尾部截断 返回所有元素
print(alist[100:])  # 切片开始位置大于列表长度时,返回空列表
alist[len(alist):] = [9]  # 在列表尾部增加元素
print(alist)
alist[:0] = [1, 2]  # 前面的0省略了,相当于是alist[0:0] = [1,2]
print(alist)      # 在列表的头部增加元素
alist[3:3] = [4]  # 在列表中间插入元素
print(alist)
alist[:3] = [1, 2]  # 相当于是alist[0:3] = [1,2] 替换列表元素
alist[3:] = [4, 5, 6]  # 替换元素,结果按照两边的最短元素来决定.
print(alist)
alist[:3] = []  # 删除列表中前三个元素 置为空
del alist[:3]     #切片元素连续
del alist[::2]     #隔一个删除一个,切片元素不连续

4. bisect处理已排序序列

import bisect
# bisect 处理(维持)已排序序列 升序
# 查找原理 二分查找

#内部方法
# insort = insort_right
# bisect = bisect_right
# insort_left
# bisect_left 

inter_lis = [] #可以定义任意可变序列

if __name__ == '__main__':

    bisect.insort(inter_lis, 3)
    bisect.insort(inter_lis, 4)
    bisect.insort(inter_lis, 1)
    bisect.insort(inter_lis, 9)
    a = bisect.bisect(inter_lis, 4)  # bisect就是bisect_right
    b = bisect.bisect_left(inter_lis, 4)
    print(b)  # 返回索引2 在left插入
    print(a)  # 返回索引3 在right插入
    print(inter_lis)  # 自动排序

🌙为什么要存在left和right两个方法:

inter_lis = [] #可以定义任意可变序列
inter_lis1 = []

#在操作 1和1.0时 内部认为两个值是相同的 这时候就需要left和right来选择插入
if __name__ == '__main__':
    
    bisect.insort(inter_lis, 1)
    bisect.insort(inter_lis, 1.0)
    print(inter_lis) #输出[1, 1.0]
    
    bisect.insort(inter_lis1, 1)
    bisect.insort_left(inter_lis1, 1.0)
    print(inter_lis1) #输出[1.0, 1]

5. 什么情况下不使用list

🌙 上述可变类型解释过array的用法,这里单独区分一下使用list和array的选择:

import array

add_list = []
add_array = array.array("I")

if __name__ == '__main__':
    add_list.extend([1, 2, "Name"])
    # add_array.extend([1, 2, "Name"]) 报错
    add_array.extend(range(3, 5))

🌙在存放不同类型的数据时选择list,在相同数据存储的情况下array的性能大于list

6. Python 生成式的写法

6.1 列表生成式

odd_list = []
odd_list = [i for i in range(21) if i % 2 == 1] #列表生成式
print(odd_list) #输出[0,21)的奇数集合

#在逻辑复杂的情况
def demo(x):
    return x*x
odd_list = [demo(x) for x in  range(21) if x % 2 == 1]

在复杂的情况 列表生成式性能高于列表操作

6.2 生成器表达式

odd_list = (i for i in range(21) if i % 2 == 1) #生成器
odd_list1 = list(odd_list) #生成list
print("奇数集合", oddlist1)

6.3 字典推导式

My_dict = {"name":"张三", "Age":19}
My_dic = {value:key for key ,value in My_dict.items()}
print("输出整个dict", My_dic)

6.4 集合推导式子

My_dict = {"name":"张三", "Age":19}
my_set ={key for key, value in My_dict.items()}
print("输出key", my_set)
my_set ={value for key, value in My_dict.items()} # items
print("输出value", my_set)
my_set = {value: key for key, value in My_dict.items()}  # items
print(type(my_set))
print("反向输出dict", my_set)
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

让我在看看

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值