徹底理解Python list切片原理
關(guān)于list的insert函數(shù)
list#insert(ind,value)在ind元素前面插入value
首先對ind進(jìn)行預(yù)處理:如果ind<0,則ind+=len(a),這樣一來ind就變成了正數(shù)下標(biāo)
預(yù)處理之后,
當(dāng)ind<0時,ind=0,相當(dāng)于頭部插入
當(dāng)ind>len(a)時,ind=len(a),相當(dāng)于尾部插入
切片實例
Python中的列表切片非常靈活,要根據(jù)表象來分析它的內(nèi)在機理,這樣用起來才能溜。
下標(biāo)可以為負(fù)數(shù)有利有弊,好處是使用起來更簡便,壞處是當(dāng)我下表越界了我也不知道反倒發(fā)生奇奇怪怪的錯誤。
print str[0:3] #截取第一位到第三位的字符 print str[:] #截取字符串的全部字符 print str[6:] #截取第七個字符到結(jié)尾 print str[:-3] #截取從頭開始到倒數(shù)第三個字符之前 print str[2] #截取第三個字符 print str[-1] #截取倒數(shù)第一個字符 print str[::-1] #創(chuàng)造一個與原字符串順序相反的字符串 print str[-3:-1] #截取倒數(shù)第三位與倒數(shù)第一位之前的字符 print str[-3:] #截取倒數(shù)第三位到結(jié)尾 print str[:-5:-3] #逆序截取
可見,列表的下標(biāo)有三個參數(shù):beg(起始下標(biāo)),end(終止下標(biāo)),delta(變化量)
- 當(dāng)delta小于0時,beg默認(rèn)為len(array)-1,end默認(rèn)為開頭之前。
- 當(dāng)delta大于0時,beg默認(rèn)為0,end默認(rèn)為最末之后。
- 當(dāng)delta未給出時:delta默認(rèn)為1
這個代碼示例演示了大概原理,學(xué)習(xí)一件事物,先學(xué)習(xí)它的表象,然后分析它的內(nèi)在實現(xiàn),最后查看源代碼仔細(xì)推敲它到底是怎么實現(xiàn)的。
需要注意的是,列表切片產(chǎn)生的是列表的副本,與原列表不是同一份空間。
x=[1,2,3] y=x[:] x[0]=-1 print(y) #輸出[1,2,3]
列表切片寫操作
接下來探究切片的寫操作
>>> x=[1,2,3,4,5] >>> x[2:0]=100 #在2后面插入若干個元素,應(yīng)該用列表 Traceback (most recent call last): File "<stdin>", line 1, in <module> TypeError: can only assign an iterable >>> x[2:0]=[100] >>> x [1, 2, 100, 3, 4, 5] >>> del x[2:3] #刪除切片 >>> x [1, 2, 3, 4, 5] >>> x[2:1]=[100] #對于切片x[from:to],會進(jìn)行預(yù)處理to=max(from+1,to) >>> x [1, 2, 100, 3, 4, 5] >>> del x[2:0] #對于切片del操作,如果from>to,不執(zhí)行任何操作 >>> x [1, 2, 100, 3, 4, 5] >>> del x[2:1] >>> x [1, 2, 100, 3, 4, 5] >>> del x[2:3] >>> x [1, 2, 3, 4, 5] >>> x[2:4]=None Traceback (most recent call last): File "<stdin>", line 1, in <module> TypeError: can only assign an iterable >>> x[2:4]=[None] >>> x [1, 2, None, 5]
對列表切片進(jìn)行深入理解:
def between(beg, end, mid): # 判斷mid是否位于begin和end之間 return end > mid >= beg or end < mid <= beg def get_slice(a, beg, end, delta=1): # 數(shù)組切片get方式 if delta == 0: raise ValueError("slice step cannot be 0") # 將負(fù)數(shù)下標(biāo)轉(zhuǎn)化一下 if beg < 0: beg += len(a) if end < 0: end += len(a) # 如果轉(zhuǎn)化完成之后依然不在合法范圍內(nèi),則返回空列表 if beg < 0 and end < 0 or beg >= len(a) and end >= len(a): return [] # 如果方向不同,則返回空列表 if (end - beg) * delta <= 0: return [] # 將越界的部分進(jìn)行裁剪 beg = max(0, min(beg, len(a) - 1)) end = max(-1, min(end, len(a))) ans = [] i = beg while between(beg, end, i): ans.append(a[i]) i += delta return ans def set_slice(a, li, beg, end, delta=1): if delta == 0: raise ValueError("slice step cannot be 0") if delta == 1: # 如果delta==1,那么li的長度可以隨意 if beg < 0: beg += len(a) if end < 0: end += len(a) beg = max(0, min(beg, len(a) - 1)) end = max(-1, min(end, len(a))) for i in range(beg, end): del a[beg] for i in reversed(li): a.insert(beg, i) else: # delta!=1,相當(dāng)于替換 if len(get_slice(a, beg, end, delta)) != len(li): raise ValueError("array don't match") if len(li) == 0: return if beg < 0: beg += len(a) if end < 0: end += len(a) beg = max(0, min(beg, len(a) - 1)) # 用li中的全部元素逐一替換 for ind, value in enumerate(li): a[ind * delta + beg] = value def test_getSlice(): a = list(range(10)) import random for i in range(10): beg = random.randint(-15, 15) end = random.randint(-15, 15) delta = 0 while delta == 0: delta = random.randint(-15, 15) print(len(get_slice(a, beg, end, delta)) == len(a[beg:end:delta]), beg, end, delta) def test_setSlice(): import random for i in range(10): a = list(range(10)) beg = random.randint(-15, 15) end = random.randint(-15, 15) delta = 0 while delta == 0: delta = random.randint(-5, 5) sz = len(a[beg:end:delta]) if delta == 1: sz = random.randint(0, 4) li = [random.randint(0, 100) for i in range(sz)] set_slice(a, li, beg, end, delta) mine = a a = list(range(10)) a[beg:end:delta] = li print(a == mine) test_setSlice()
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Python入門之實例方法、類方法和靜態(tài)方法的區(qū)別講解
這篇文章主要介紹了Python入門之實例方法、類方法和靜態(tài)方法的區(qū)別講解,實例方法是在創(chuàng)建了類的實例之后才能被調(diào)用的方法,類方法是在不需要創(chuàng)建類的實例的情況下就可以調(diào)用的方法,最后,靜態(tài)方法是與類和類的實例都沒有綁定關(guān)系的方法,需要的朋友可以參考下2023-10-10Django中數(shù)據(jù)在前后端傳遞的方式之表單、JSON與ajax
Django從后臺往前臺傳遞數(shù)據(jù)時有多種方法可以實現(xiàn),下面這篇文章主要給大家介紹了關(guān)于Django中數(shù)據(jù)在前后端傳遞的方式之表單、JSON與ajax的相關(guān)資料,需要的朋友可以參考下2022-10-1030秒學(xué)會30個超實用Python代碼片段【收藏版】
許多人在數(shù)據(jù)科學(xué)、機器學(xué)習(xí)、web開發(fā)、腳本編寫和自動化等領(lǐng)域中都會使用Python,它是一種十分流行的語言。本文將簡要介紹30個簡短的、且能在30秒內(nèi)掌握的代碼片段,感興趣的朋友一起看看吧2019-10-10tesserocr與pytesseract模塊的使用方法解析
這篇文章主要介紹了tesserocr與pytesseract模塊的使用方法解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2019-08-08Python遍歷指定文件夾下的所有文件名的方法小結(jié)
當(dāng)需要遍歷指定文件夾下的所有文件名時,Python提供了多種方法來實現(xiàn)這個任務(wù),本文將介紹如何使用Python來完成這一任務(wù),有需要的小伙伴可以參考下2024-01-01