整個(gè)九月份,我都在忙一個(gè)js的項(xiàng)目。因?yàn)楹枚嗄瓴粚?xiě)js,動(dòng)手之前特地找了一些js的資料惡補(bǔ)半天,結(jié)果發(fā)現(xiàn)js越來(lái)越像Python了。且不說(shuō)js從基于原型的面向?qū)ο筠D(zhuǎn)向了基于類(lèi)的面向?qū)ο螅瑔问穷?lèi)型化數(shù)組(Typed Arrays)的引入,就讓用慣了NumPy的我喜出望外。另外,js的數(shù)組推導(dǎo)式和裝飾器,也幾乎是完全照抄了Python的作業(yè)。
進(jìn)入十月,Python社區(qū)指導(dǎo)委員會(huì)正式推出了Python3.10,距離上一個(gè)版本發(fā)布,正好過(guò)去了一年。記得Python3.9發(fā)布之后沒(méi)幾天,我寫(xiě)過(guò)一篇名為《危險(xiǎn)的轉(zhuǎn)變:Python正在從簡(jiǎn)明轉(zhuǎn)向臃腫,從實(shí)用轉(zhuǎn)向媚俗》的博文,著實(shí)吐槽了一番,正所謂愛(ài)之也深恨之也切。這次新版本發(fā)布,我自然不會(huì)放過(guò)嘗鮮的機(jī)會(huì),趕緊安裝嘗試了一下,卻發(fā)現(xiàn),一向特立獨(dú)行的Python居然開(kāi)始抄作業(yè)了,這次抄的是Rust。
眾所周知,Rust有兩樣鎮(zhèn)山之寶,一是安全的內(nèi)存模型,二是模式匹配。在內(nèi)存管理上,Python使用了傳統(tǒng)的垃圾回收的內(nèi)存模型,和Rust沒(méi)有多少可比性。隨著Python3.10的發(fā)布,模式匹配被引入到Python中,而且?guī)缀跏峭耆瞻崃薘ust的概念。Rust支持模式匹配中的變量綁定、結(jié)構(gòu)體/元組解構(gòu)、守衛(wèi)條件判斷、數(shù)值范圍匹配等特性,Python照單全收,連下劃線 _ 匹配任意情形也原封不動(dòng)地繼承了過(guò)來(lái)。
讓我們一起來(lái)揭開(kāi)Pyhton3.10最重要的升級(jí)——模式匹配的蓋頭。
類(lèi)似C語(yǔ)言的switch case, Python的模式匹配最簡(jiǎn)單的應(yīng)用就是對(duì)字面值進(jìn)行匹配:
- >>>a=3
- >>>match(a):
- case1:
- print("a==1")
- case2:
- print("a==2")
- case_:#default
- print("other")
- other
case語(yǔ)句中,支持或操作:
- >>>importdatetime
- >>>n=datetime.datetime.now()
- >>>match(n.weekday()):
- case0|1|2|3|4:print("工作日")
- case5|6:print("周末")
- 工作日
除了字面值外,case語(yǔ)句,支持對(duì)上面提到的模式進(jìn)行解構(gòu),如對(duì)元組:
- >>>a=(0,1)
- >>>match(a):
- case(0,y):#匹配所有第0個(gè)元素是0的元組
- print(f"a[0]==0,a[1]=={y}")
- case(x,0):#匹配所有第1個(gè)元素是0的元組
- print(f"a[1]==0,a[0]=={x}")
- a[0]==0,a[1]==1
對(duì)列表:
- >>>cmd="lstest"
- >>>match(cmd.split()):
- case["ls",path]:print(f"顯示{path}中的文件和目錄")
- case["rm",path]:print(f"刪除{path}中的文件和目錄")
- case["cp",src,dest]:print(f"將{src}復(fù)制到{dest}")
- 顯示test中的文件和目錄
對(duì)字典:
- >>>a={"name":"xxx","age":40,"job":"程序員"}
- >>>match(a):
- case{"name":name,"age":age,"job":"程序員"}:
- print(f"他是一名程序員,名字叫{name},{age}歲了")
- case{"name":name,"age":age,"job":"教師"}:
- print(f"他是一名人民教師,名字叫{name},{age}歲了")
- 他是一名程序員,名字叫xxx,40歲了
對(duì)于類(lèi)對(duì)象,match case照樣可以使用如:
- >>>classPoint():
- def__init__(self,x,y):
- self.x=x
- self.y=y
- >>>a=Point(1,2)
- >>>match(a):
- casePoint(x=1,y=y):print(f"這是一個(gè)X坐標(biāo)為1的點(diǎn),它的Y坐標(biāo)為{y}")
- casePoint(x=x,y=2):print(f"這是一個(gè)Y坐標(biāo)為2的點(diǎn),它的X坐標(biāo)為{x}")
- 這是一個(gè)X坐標(biāo)為1的點(diǎn),它的Y坐標(biāo)為2
也可以用于多個(gè)類(lèi):
- >>>classProgrammer:
- def__init__(self,lang):
- self.lang=lang
- >>>classTeacher:
- def__init__(self,subject):
- self.subject=subject
- >>>a=Programmer("Python")
- >>>match(a):
- caseProgrammer(lang="Python"):print("咱們都是Pyhon程序員!")
- caseProgrammer():print("原來(lái)你也是一名程序員!")
- caseTeacher():print("向人民教師致敬!")
- 咱們都是Pyhon程序員!
case 語(yǔ)句后,還支持添加一個(gè)if語(yǔ)句,進(jìn)一步對(duì)匹配的條件進(jìn)行限制,這個(gè)if語(yǔ)句,被稱(chēng)之為“守衛(wèi)”。如:
- >>>classPoint():
- def__init__(self,x,y):
- self.x=x
- self.y=y
- >>>a=Point(2,2)
- >>>match(a):
- casePoint(x=x,y=y)ifx==y:print("這個(gè)點(diǎn)在斜率為1的直線上")
- casePoint(x=x,y=y)ifx==-y:print("這個(gè)點(diǎn)在斜率為-1的直線上")
- 這個(gè)點(diǎn)在斜率為1的直線上
美中不足的是,我沒(méi)有找到case語(yǔ)句中直接使用范圍的方法,但這個(gè)可以用守衛(wèi)來(lái)解決:
- >>>a=5
- >>>match(a):
- casexif1<=x<10:print("數(shù)字在1和10之間")
- casexif10<=x<20:print("數(shù)字在10和20之間")
- 數(shù)字在1和10之間
原文鏈接:https://mp.weixin.qq.com/s/kXblwnxsphb00ATDPfsdTA