自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

魔法方法推開Python進(jìn)階學(xué)習(xí)大門

開發(fā) 后端
Python是Guido van Rossum設(shè)計(jì)出來(lái)的讓使用者覺得如沐春風(fēng)的一門編程語(yǔ)言。

[[376059]]

本文轉(zhuǎn)載自微信公眾號(hào)「PythonMind」,作者dongfanger。轉(zhuǎn)載本文請(qǐng)聯(lián)系PythonMind公眾號(hào)。  

熱愛Python

Python是Guido van Rossum設(shè)計(jì)出來(lái)的讓使用者覺得如沐春風(fēng)的一門編程語(yǔ)言。2020年11月12日,64歲的Python之父宣布由于退休生活太無(wú)聊,自己決定加入Microsoft的DevDiv Team,致力于“確保更好地使用Python”。盡管在國(guó)內(nèi)有些聲音在Diss著Python,認(rèn)為它太簡(jiǎn)單,只是個(gè)腳本語(yǔ)言,但是它的發(fā)明者對(duì)Python的熱情,仍然激勵(lì)著我們堅(jiān)持對(duì)Python的熱愛。

龜叔是所有編程語(yǔ)言發(fā)明者當(dāng)中頭發(fā)最多的這位。

[[376060]]

 

奇跡時(shí)刻

collection.len()是面向?qū)ο笳Z(yǔ)言的寫法,len(collection)是Python語(yǔ)言的寫法,這種風(fēng)格叫做Pythonic。從前者到后者,就像變魔術(shù)一樣,一瞬間讓人眼前一亮。這個(gè)魔術(shù)就是Python魔法方法,或者叫雙下方法,它是用雙下劃線開頭和雙下劃線結(jié)尾的特殊方法,比如obj[key],Python解釋器實(shí)際上會(huì)轉(zhuǎn)換成obj.__getitem__(key)來(lái)運(yùn)行,但是使用者并無(wú)感知。

__getitem__和__len

____getitem__用來(lái)獲取數(shù)據(jù),__len__用來(lái)返回長(zhǎng)度,這2個(gè)魔法方法是Python基礎(chǔ),我們通過一副撲克牌來(lái)了解:

  1. import collections 
  2.  
  3. # 定義一副牌 
  4. Card = collections.namedtuple('Card', ['rank''suit']) 
  5.  
  6. class FrenchDeck: 
  7.     # 大小 
  8.     ranks = [str(n) for n in range(2, 11)] + list('JQKA'
  9.     # 花色 
  10.     suits = 'spades diamonds clubs hearts'.split() 
  11.  
  12.     def __init__(self): 
  13.         # 生成一副牌 
  14.         self._cards = [Card(rank, suit) for suit in self.suits 
  15.                                         for rank in self.ranks] 
  16.  
  17.     def __len__(self): 
  18.         return len(self._cards) 
  19.  
  20.     def __getitem__(self, position): 
  21.         return self._cards[position] 

本來(lái)我們對(duì)這副牌什么都不能做,但是由于實(shí)現(xiàn)了__len__,可以使用len()函數(shù)查看有多少?gòu)埮疲?/p>

  1. >>> len(deck) 
  2. 52 

由于實(shí)現(xiàn)了__getitem__,可以使用中括號(hào)索引取值:

  1. >>> deck[0] 
  2. Card(rank='2', suit='spades'

能進(jìn)行切片:

  1. >>> deck[:3] 
  2. [Card(rank='2', suit='spades'), Card(rank='3', suit='spades'), Card(rank='4', suit='spades')] 
  3. >>> deck[12::13] 
  4. [Card(rank='A', suit='spades'), Card(rank='A', suit='diamonds'), Card(rank='A', suit='clubs'), Card(rank='A', suit='hearts')] 

能迭代:

  1. >>> for card in deck:  # doctest: +ELLIPSIS 
  2. ...   print(card) 
  3. Card(rank='2', suit='spades'
  4. Card(rank='3', suit='spades'
  5. Card(rank='4', suit='spades'
  6. ... 

發(fā)現(xiàn)沒有,魔法方法是可以用來(lái)裝B的!別人寫個(gè)類只能get、set,你寫個(gè)類還能花式炫技,666。

Python魔法方法是給Python解釋器使用的,一般不需要直接調(diào)用,Python會(huì)自己去調(diào),比如把len(my_object)寫成my_object.__len__(),就弄巧成拙了。

魔法方法實(shí)現(xiàn)運(yùn)算符

前面例子實(shí)現(xiàn)了取值和長(zhǎng)度,接著再看一個(gè)例子,使用__repr__、__abs__、__bool__、__add__、__mul__,實(shí)現(xiàn)運(yùn)算符:

  1. from math import hypot 
  2.  
  3. # 二維向量 
  4. class Vector: 
  5.  
  6.     def __init__(self, x=0, y=0): 
  7.         self.x = x 
  8.         self.y = y 
  9.  
  10.     # 表達(dá)式 
  11.     def __repr__(self): 
  12.         return 'Vector(%r, %r)' % (self.x, self.y) 
  13.  
  14.     # 絕對(duì)值 
  15.     def __abs__(self): 
  16.         return hypot(self.x, self.y) 
  17.  
  18.     # 布爾值 
  19.     def __bool__(self): 
  20.         return bool(abs(self)) 
  21.  
  22.     # 加法 
  23.     def __add__(self, other): 
  24.         x = self.x + other.x 
  25.         y = self.y + other.y 
  26.         return Vector(x, y) 
  27.  
  28.     #乘法 
  29.     def __mul__(self, scalar): 
  30.         return Vector(self.x * scalar, self.y * scalar) 

__add__實(shí)現(xiàn)了加法:

  1. >>> v1 = Vector(2, 4) 
  2. >>> v2 = Vector(2, 1) 
  3. >>> v1 + v2 
  4. Vector(4, 5) 

__abs__實(shí)現(xiàn)了絕對(duì)值:

  1. >>> v = Vector(3, 4) 
  2. >>> abs(v) 
  3. 5.0 

__mul__實(shí)現(xiàn)了乘法:

  1. >>> v * 3 
  2. Vector(9, 12) 

__repr__實(shí)現(xiàn)了對(duì)象的字符串表示:

  1. Vector(4, 5) 

否則得到的字符串可能是。

__bool__實(shí)現(xiàn)了布爾值:

  1. if Vector(4, 5): 
  2.     return True 

其他魔法方法

 

 


一篇文章是講不完魔法方法的,我們會(huì)在后續(xù)文章中,繼續(xù)探討如何使用和實(shí)現(xiàn)它們。

 

 

Tips

本小節(jié)內(nèi)容是我看《流暢的Python》第一遍時(shí)記錄的知識(shí)點(diǎn):

  1. collections.namedtuple可以用來(lái)創(chuàng)建只有少數(shù)屬性但沒有方法的對(duì)象,比如
  1. beer_card = Card('7''diamonds'
  1. 2.random.choice和random.sample不一樣的地方在于,sample是返回序列,choice是返回元素,當(dāng)使用sample(list, 1)[0]的時(shí)候,不如直接使用choice(list)。
  2. 特殊方法的存在是為了被Python解釋器調(diào)用的。
  3. PyVarObject是表示內(nèi)存中長(zhǎng)度可變的內(nèi)置對(duì)象的C語(yǔ)言結(jié)構(gòu)體。list或str或bytearray的__len__實(shí)際上返回的PyVarObject.ob_size屬性,這個(gè)比調(diào)用一個(gè)方法要快的多。
  4. len之所以不是一個(gè)普通方法,是為了讓python自帶的數(shù)據(jù)結(jié)構(gòu)可以走后門,abs也是同理。
  5. 很多時(shí)候調(diào)用__init__方法的目的是,在你自己的子類的__init__方法中調(diào)用超類的構(gòu)造器。
  6. abs,如果輸入是整數(shù)或者浮點(diǎn)數(shù),它返回的是輸入值的絕對(duì)值;如果輸入是復(fù)數(shù),那么返回這個(gè)復(fù)數(shù)的模。
  7. __repr__和__str__二選一的話,__repr__更好,因?yàn)槿绻粋€(gè)對(duì)象沒有__str__函數(shù),解釋器會(huì)用__repr__作為替代。
  8. python對(duì)象的一個(gè)基本要求就是它得有合理的字符串表示形式,這就是數(shù)據(jù)模型中存在特殊方法__repr__和__str__的原因。
  9. 為了判定一個(gè)值x為真還是為假,python會(huì)調(diào)用bool(x),它的背后是調(diào)用x.__bool__()。如果不存在,就會(huì)調(diào)用x.__len__(),返回0為Flase,非0為True。
  10. python通過運(yùn)算符重載這一模式提供了豐富的數(shù)值類型,除了內(nèi)置那些,還有decimal.Decimal和fractions.Fraction。

 

責(zé)任編輯:武曉燕 來(lái)源: PythonMind
相關(guān)推薦

2022-12-19 15:12:34

python運(yùn)算符

2022-07-07 08:01:51

Python魔法方法代碼

2017-04-18 10:59:28

軟件測(cè)試認(rèn)證

2024-08-19 10:21:37

接口Python魔法方法

2020-04-03 16:52:01

基建數(shù)字發(fā)展

2022-05-10 09:19:29

異網(wǎng)漫游5G網(wǎng)絡(luò)

2023-07-04 08:33:46

Python對(duì)象編程

2010-02-05 11:23:01

C++聲明語(yǔ)法

2022-08-08 15:48:17

研究模型

2024-10-29 10:54:07

2022-11-04 17:02:31

AI模型

2016-03-31 14:07:51

大數(shù)據(jù)數(shù)據(jù)可視化

2015-04-16 13:38:26

GPU計(jì)算深度學(xué)習(xí)NVIDIA

2024-12-04 13:00:00

2020-09-18 06:48:21

Python編程語(yǔ)言

2010-07-23 14:34:38

2020-01-19 11:57:44

Python 開發(fā)編程語(yǔ)言

2013-04-16 09:30:08

OpenStack開源云

2021-11-29 07:02:24

Python函數(shù)操作

2022-01-11 18:20:42

AndroidRoadMap程序
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)