選取了 6 個(gè)自己認(rèn)為值得玩味的 python 代碼,希望對(duì)正在學(xué)習(xí) python 的你有所幫助。
1、類有兩個(gè)方法,一個(gè)是 __new__,一個(gè)是 __init__,有什么區(qū)別,哪個(gè)會(huì)先執(zhí)行呢?
classtest(object):
def__init__(self):
print("test->__init__")
def__new__(cls):
print("test->__new__")
returnsuper().__new__(cls)
a=test()
運(yùn)行結(jié)果如下:
test->__new__
test->__init__
再來看另一個(gè)例子
classtest2(object):
def__init__(self):
print("test2->__init__")
def__new__(cls):
print("test2->__new__")
returnobject()
b=test2()
運(yùn)行結(jié)果如下:
test2->__new__
這里給出官方的解釋:__init__ 作用是類實(shí)例進(jìn)行初始化,第一個(gè)參數(shù)為 self,代表對(duì)象本身,可以沒有返回值。__new__ 則是返回一個(gè)新的類的實(shí)例,第一個(gè)參數(shù)是 cls 代表該類本身,必須有返回值。很明顯,類先實(shí)例化才能產(chǎn)能對(duì)象,顯然是 __new__ 先執(zhí)行,然后再 __init__,實(shí)際上,只要 __new__ 返回的是類本身的實(shí)例,它會(huì)自動(dòng)調(diào)用 __init__ 進(jìn)行初始化。但是有例外,如果 __new__ 返回的是其他類的實(shí)例,則它不會(huì)調(diào)用當(dāng)前類的 __init__。下面我們分別輸出下對(duì)象 a 和對(duì)象 b 的類型:
print(type(a))
#
print(type(b))
#'object'>'__main__.test'>
可以看出,a 是 test 類的一個(gè)對(duì)象,而 b 就是 object 的對(duì)象。
參考文檔:
https://docs.python.org/3/reference/datamodel.html?highlight=__new__#object.__new__
2、map 函數(shù)返回的對(duì)象
map()函數(shù)第一個(gè)參數(shù)是 fun,第二個(gè)參數(shù)是一般是 list,第三個(gè)參數(shù)可以寫 list,也可以不寫,作用就是對(duì)列表中 list 的每個(gè)元素順序調(diào)用函數(shù) fun 。
>>>b=map(lambdax:x*x,[1,2,3])
>>>[iforiinb]
[1,4,9]
>>>[iforiinb]
[]
>>>
有沒有發(fā)現(xiàn),第二次輸出 b 中的元素時(shí),發(fā)現(xiàn)變成空了。原因是 map() 函數(shù)返回的是一個(gè)迭代器,并用對(duì)返回結(jié)果使用了 yield,這樣做的目的在于節(jié)省內(nèi)存。
舉個(gè)例子:
#encoding:UTF-8
defyield_test(n):
foriinrange(n):
yieldcall(i)
#做一些其它的事情
defcall(i):
returni*2
#使用for循環(huán)
x=yield_test(5)
print([iforiinx])
print([iforiinx])
執(zhí)行結(jié)果為:
[0,2,4,6,8]
[]
這里如果不用 yield,那么在列表中的元素非常大時(shí),將會(huì)全部裝入內(nèi)存,這是非常浪費(fèi)內(nèi)存的,同時(shí)也會(huì)降低效率。
3、正則表達(dá)式中 compile 是否多此一舉?
比如現(xiàn)在有個(gè)需求,對(duì)于文本
中國(guó)
,用正則匹配出標(biāo)簽里面的“中國(guó)”,其中 class 的類名是不確定的。有兩種方法,代碼如下:
>>>importre
>>>text='中國(guó)'
>>>#方法一
...
>>>re.findall('(.*)',text)
['中國(guó)']
>>>#方法二
...
>>>regex='(.*)'
>>>pattern=re.compile(regex)
>>>re.findall(pattern,text)
['中國(guó)']
>>>
這里為什么要用 compile 多寫兩行代碼呢?原因是 compile 將正則表達(dá)式編譯成一個(gè)對(duì)象,加快速度,并重復(fù)使用。
4、[[1,2],[3,4],[5,6]]一行代碼展開該列表,得出[1,2,3,4,5,6]
>>>[jforiin[[1,2],[3,4],[5,6]]forjini]
[1,2,3,4,5,6]
>>>
5、一行代碼將字符串 "->" 插入到 "abcdefg"中每個(gè)字符的中間
>>>"->".join("abcdef")
'a->b->c->d->e->f'
>>>
這里也建議多使用 os.path.join() 來拼接操作系統(tǒng)的文件路徑。
6、zip 函數(shù)
zip() 函數(shù)在運(yùn)算時(shí),會(huì)以一個(gè)或多個(gè)序列(可迭代對(duì)象)做為參數(shù),返回一個(gè)元組的列表。同時(shí)將這些序列中并排的元素配對(duì)。zip() 參數(shù)可以接受任何類型的序列,同時(shí)也可以有兩個(gè)以上的參數(shù);當(dāng)傳入?yún)?shù)的長(zhǎng)度不同時(shí),zip 能自動(dòng)以最短序列長(zhǎng)度為準(zhǔn)進(jìn)行截取,獲得元組。
>>>a=[1,2]
>>>b=(3,4)
>>>zip(a,b)
>>>foriinzip(a,b):
...print(i)
...
(1,3)
(2,4)
>>>a="ab"
>>>b="xyz"
>>>foriinzip(a,b):
...print(i)
...
('a','x')
('b','y')
>>>objectat0x000001a20201aa08>
審核編輯:湯梓紅
-
函數(shù)
+關(guān)注
關(guān)注
3文章
4346瀏覽量
62970 -
代碼
+關(guān)注
關(guān)注
30文章
4825瀏覽量
69046 -
python
+關(guān)注
關(guān)注
56文章
4807瀏覽量
85037
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論