python中的序列分類
序列類型的分類:
① 容器序列:list,tuple,deque(可以防任意的類型的容器)
② 扁平序列:str,bytes,bytearray,array.array(可以使用 for循環遍歷的)
③ 可變序列:list,deque,bytearray,array
④ 不可變:str,tuple,bytes
python中序列類型的abc繼承關系
from collections import abc
查看abc的源碼
__all__ = ["Awaitable", "Coroutine",
"AsyncIterable", "AsyncIterator", "AsyncGenerator",
"Hashable", "Iterable", "Iterator", "Generator", "Reversible",
"Sized", "Container", "Callable", "Collection",
"Set", "MutableSet",
"Mapping", "MutableMapping",
"MappingView", "KeysView", "ItemsView", "ValuesView",
"Sequence", "MutableSequence",
"ByteString",
]
Sequence不可變序列
class Sequence(Reversible, Collection):
"""All the operations on a read-only sequence.
Concrete subclasses must override __new__ or __init__,
__getitem__, and __len__.
"""
__slots__ = ()
可以看到 Sequence 繼承 Reversible和Collection
Reversible可以實現反轉
class Reversible(Iterable):
__slots__ = ()
@abstractmethod
def __reversed__(self):
while False:
yield None
Collection 繼承 Sized, Iterable, Container
class Collection(Sized, Iterable, Container): pass
Sized 可以使用len()查看對象的長度
class Sized(metaclass=ABCMeta):
__slots__ = ()
@abstractmethod
def __len__(self):
return 0
Iterable 可以是一個對象變成可迭代對象
class Iterable(metaclass=ABCMeta):
__slots__ = ()
@abstractmethod
def __iter__(self):
while False:
yield None
Container 可以是if in 判斷一個屬性是否在對象內
class Container(metaclass=ABCMeta):
__slots__ = ()
@abstractmethod
def __contains__(self, x):
return False
MutableSequence可變序列
繼承 Sequence,自己內部也封裝了增刪改查的方法
class MutableSequence(Sequence): __slots__ = () """All the operations on a read-write sequence. Concrete subclasses must provide __new__ or __init__, __getitem__, __setitem__, __delitem__, __len__, and insert(). """ @abstractmethod def __setitem__(self, index, value): raise IndexError @abstractmethod def __delitem__(self, index): raise IndexError @abstractmethod def insert(self, index, value): 'S.insert(index, value) -- insert value before index' raise IndexError def append(self, value): 'S.append(value) -- append value to the end of the sequence' self.insert(len(self), value) def clear(self): 'S.clear() -> None -- remove all items from S' try: while True: self.pop() except IndexError: pass def reverse(self): 'S.reverse() -- reverse *IN PLACE*' n = len(self) for i in range(n//2): self[i], self[n-i-1] = self[n-i-1], self[i] def extend(self, values): 'S.extend(iterable) -- extend sequence by appending elements from the iterable' for v in values: self.append(v) def pop(self, index=-1): '''S.pop([index]) -> item -- remove and return item at index (default last). Raise IndexError if list is empty or index is out of range. ''' v = self[index] del self[index] return v def remove(self, value): '''S.remove(value) -- remove first occurrence of value. Raise ValueError if the value is not present. ''' del self[self.index(value)] def __iadd__(self, values): self.extend(values) return self
Sequence 繼承 Reversible, Collection 功能和不可變序列一樣
class Sequence(Reversible, Collection):
pass
list中+,+= 和extend方法區別
’+’兩邊必須為統一類型的
a=[1,2] b=a+[3,4] # 這樣是可以的=>b=[1,2,3,4] b=a+(3,4) # 這樣是不可行的,’+’兩邊必須為統一類型的
+= 只要是可迭代對象就可以想加
c=[3,4] c += (1,2) # 這樣也是可以的 =>c=[3,4,1,2] c += ‘hello’ #這樣也是可以的 =>c=[3,4,’h’,’e’,’l’,’l’,’o’]
內部實現
會調用__iadd__
def __iadd__(self, values):
self.extend(values)
return self
extend 遍歷添加
def extend(self, values):
'S.extend(iterable) -- extend sequence by appending elements from the iterable'
for v in values:
self.append(v)
extend 直接作用域本身,沒返回值
d=[5,6] a.entend(d) #這樣是可以的,a=[1,2,5,6] d=(5,6) a.extend(d) #這樣也是可以的,a=[1,2,5,6]
實現可切片的對象
python 幫我們實現的切片
# 模式[start:end:step]
"""
其中,第一個數字start表示切片開始位置,默認為0;
第二個數字end表示切片截止(但不包含)位置(默認為列表長度);
第三個數字step表示切片的步長(默認為1)。
當start為0時可以省略,當end為列表長度時可以省略,
當step為1時可以省略,並且省略步長時可以同時省略最后一個冒號。
另外,當step為負整數時,表示反向切片,這時start應該比end的值要大才行。
"""
aList = [3, 4, 5, 6, 7, 9, 11, 13, 15, 17]
print(aList[::]) # 返回包含原列表中所有元素的新列表
print(aList[::-1]) # 返回包含原列表中所有元素的逆序列表
print(aList[::2]) # 隔一個取一個,獲取偶數位置的元素
print(aList[1::2]) # 隔一個取一個,獲取奇數位置的元素
print(aList[3:6]) # 指定切片的開始和結束位置
aList[0:100] # 切片結束位置大於列表長度時,從列表尾部截斷
aList[100:] # 切片開始位置大於列表長度時,返回空列表
aList[len(aList):] = [9] # 在列表尾部增加元素
aList[:0] = [1, 2] # 在列表頭部插入元素
aList[3:3] = [4] # 在列表中間位置插入元素
aList[:3] = [1, 2] # 替換列表元素,等號兩邊的列表長度相等
aList[3:] = [4, 5, 6] # 等號兩邊的列表長度也可以不相等
aList[::2] = [0] * 3 # 隔一個修改一個
print(aList)
aList[::2] = ['a', 'b', 'c'] # 隔一個修改一個
aList[::2] = [1, 2] # 左側切片不連續,等號兩邊列表長度必須相等
aList[:3] = [] # 刪除列表中前3個元素
del aList[:3] # 切片元素連續
del aList[::2] # 切片元素不連續,隔一個刪一個
可以看出python的切片功能是相當強大的,那我們如何去實現呢?
在這里我實現的是不可變序列 ,查看abc中 Sequence 源碼按照python協議實現其內部的魔法函數即可
import numbers
class Group:
#支持切片操作
def __init__(self, group_name, company_name, staffs):
self.group_name = group_name
self.company_name = company_name
self.staffs = staffs
def __reversed__(self):
self.staffs.reverse()
def __getitem__(self, item):
"返回一個對象"
cls = type(self)
if isinstance(item, slice):
return cls(group_name=self.group_name, company_name=self.company_name, staffs=self.staffs[item])
elif isinstance(item, numbers.Integral):
return cls(group_name=self.group_name, company_name=self.company_name, staffs=[self.staffs[item]])
def __len__(self):
return len(self.staffs)
def __iter__(self):
return iter(self.staffs)
def __contains__(self, item):
if item in self.staffs:
return True
else:
return False
staffs = ["bobby1", "imooc", "bobby2", "bobby3"]
group = Group(company_name="imooc", group_name="user", staffs=staffs)
print(group[0]) # __getitem__ 中的item 為 int 0
print(group[:2]) # __getitem__ 中的item 為 slice 類型 slice(None, 2, None)
reversed(group)
for user in group:
print(user)
輸出結果如下

bisect維護已排序序列
我們可以使用bisect 幫助我們在不斷的向序列中插入數據是,而我們的列表還是一個已排序的
import bisect #用來處理已排序的序列,用來維持已排序的序列, 升序 #二分查找 inter_list = [] bisect.insort(inter_list, 3) bisect.insort(inter_list, 2) bisect.insort(inter_list, 5) bisect.insort(inter_list, 1) bisect.insort(inter_list, 6) print(inter_list)
輸出結果如下

也可以使用雙端隊列
import bisect from collections import deque #用來處理已排序的序列,用來維持已排序的序列, 升序 #二分查找 inter_list = deque() bisect.insort(inter_list, 3) bisect.insort(inter_list, 2) bisect.insort(inter_list, 5) bisect.insort(inter_list, 1) bisect.insort(inter_list, 6) #學習成績 print(inter_list)
輸出結果

什么時候我們不該使用列表
list 和array的區別
1 list中可以存放任意的數據類型,array中只能存放單一的數據類型。
2 如果是單一的數據類型array是比list更加高校

基本使用
# array, deque
# 數組
import array
#array和list的一個重要區別, array只能存放指定的數據類型
my_array = array.array("i") # 只能存放int類型
my_array.append(1)
print(my_array)
my_array.append("abc") # 會報錯
輸出結果如下

列表推導式、生成器表達式、字典推導式
列表推導式
qu_list = [item * item for item in range(6)] print(qu_list)

#笛卡爾積 int_list1 = [1,2] int_list2 = [3,4] qu_list = [(first, second) for first in int_list1 for second in int_list2] print(qu_list)

def trs_str(item):
return str(item)
qu_list = [trs_str(item) for item in range(6)]
print(qu_list)

字典推導式
def process_item(item):
return str(item)
int_dict = {process_item(item):item for item in range(5)}
print(int_dict)

生成器表達式
my_dict = {
"key1":"bobby1",
"key2":"bobby2"
}
res = ((key, value) for key, value in my_dict.items())
print(res)

