日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程语言 > python >内容正文

python

Python中的魔法方法

發(fā)布時間:2023/12/10 python 17 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Python中的魔法方法 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

1、什么是魔法方法?

? ?  魔法方法就是可以給你的類增加魔力的特殊方法,如果你的對象實(shí)現(xiàn)(重載)了這些方法中的某一個,那么這個方法就會在特殊的情況下被 Python 所調(diào)用,你可以定義自己想要的行為,而這一切都是自動發(fā)生的。它們經(jīng)常是兩個下劃線包圍來命名的(比如 __init__,__lt__),Python的魔法方法是非常強(qiáng)大的,所以了解其使用方法也變得尤為重要!

2、__init__(self[, ...]),__new__(cls[, ...]),__del__(self)

  __init__ 構(gòu)造器,當(dāng)一個實(shí)例被創(chuàng)建的時候初始化的方法。但是它并不是實(shí)例化調(diào)用的第一個方法,__new__才是實(shí)例化對象調(diào)用的第一個方法,它只取下 cls參數(shù),并把其他參數(shù)傳給 __init__。 __new__很少使用,但是也有它適合的場景,尤其是當(dāng)類繼承自一個像元組或者字符串這樣不經(jīng)常改變的類型的時候。

  __new__ 使用時注意以下四點(diǎn):

1

2

3

4

5

1.?__new__ 是在一個對象實(shí)例化的時候所調(diào)用的第一個方法

2.?它的第一個參數(shù)是這個類,其他的參數(shù)是用來直接傳遞給 __init__ 方法

3.?__new__ 決定是否要使用該 __init__ 方法,因?yàn)?__new__ 可以調(diào)用其他類的構(gòu)造方法或者直接返回別的實(shí)例對象來作為本類的實(shí)例,如果 __new__ 沒有返回實(shí)例對象,則 __init__ 不會被調(diào)用

4.?__new__ 主要是用于繼承一個不可變的類型比如一個?tuple?或者 string

5.?__new__?return的是一個構(gòu)建的實(shí)例

  

  __new__實(shí)現(xiàn)單例模式:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

class?Person(object):

????def?__init__(self, name, age):

????????self.name?=?name

????????self.age?=?age

?

????def?__new__(cls,?*args,?**kwargs):

????????if?not?hasattr(cls,'instance'):

????????????cls.instance?=?super(Person,cls).__new__(cls)

????????return?cls.instance

?

?

a?=?Person('p1',20)

b?=?Person('p2',21)

print(a?==?b)

#這里的打印結(jié)果是True,可見a,和b都是同一個實(shí)例

#單例作用:

#第一、控制資源的使用,通過線程同步來控制資源的并發(fā)訪問;

#第二、控制實(shí)例產(chǎn)生的數(shù)量,達(dá)到節(jié)約資源的目的。

#第三、作為通信媒介使用,也就是數(shù)據(jù)共享,它可以在不建立直接關(guān)聯(lián)的條件下,讓多個不##相關(guān)的兩個線程或者進(jìn)程之間實(shí)現(xiàn)通信。

#比如,數(shù)據(jù)庫連接池的設(shè)計(jì)一般采用單例模式,數(shù)據(jù)庫連接是一種數(shù)據(jù)庫資源。

  __del__ 析構(gòu)器,當(dāng)實(shí)例被銷毀時調(diào)用

3、__call__(self[,args ...]),__getitem__(self,key),__setitem__(self,key,value)

  __call__ 允許一個類的實(shí)例像函數(shù)一樣被調(diào)用

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

class?Person(object):

????def?__init__(self, name, age):

????????self.name?=?name

????????self.age?=?age

????????self.instance?=?add

?

????def?__call__(self,*args):

????????return?self.instance(*args)

?

def?add(args):

????return?args[0]?+?args[1]

?

a?=?Person('p1',?20)

print(a([1,2]))

#這里將打印 3

#可見當(dāng)創(chuàng)建a這個對象之后,如果定義了__call__函數(shù)則對象是可以像函數(shù)一樣調(diào)用的

  __getitem__ 定義獲取容器中指定元素的行為,相當(dāng)于self[key]

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

class?Person(object):

????def?__init__(self, name, age):

????????self.name?=?name

????????self.age?=?age

????????self._registry?=?{

????????????'name': name,

????????????'age': age

????????}

?

????def?__call__(self,?*args):

????????return?self.instance(*args)

?

????def?__getitem__(self, key):

????????if?key?not?in?self._registry.keys():

????????????raise?Exception('Please registry the key:%s first !'?%?(key,))

????????return?self._registry[key]

?

a?=?Person('p1',?20)

print(a['name'],a['age'])

#這里打印的是 'p1' 20

#可見__getitem__使實(shí)例可以像字典一樣訪問

  __setitem__ 設(shè)置容器中指定元素的行為,相當(dāng)于self[key] = value?

4、__getattr__(self,name),__getattribute__(self,name),__setattr__(self,name,value),__delattr__(self,name),__get__(self,instance,owner),__set__(self,instance,value),__delete__(self,instance)

  __getattr__ 定義當(dāng)用戶試圖訪問一個不存在屬性的時候的行為

  __setattr__ 定義當(dāng)一個屬性被設(shè)置的時候的行為

  __getattribute__ 定義當(dāng)一個屬性被訪問的時候的行為

  

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

class?Person(object):

????def?__init__(self, name, age):

????????self.name?=?name

????????self.age?=?age

????????self._registry?=?{

????????????'name': name,

????????????'age': age

????????}

?

????def?__getattr__(self, item):

????????print("don't have the attribute ",item)

????????return?False

?

????def?__setattr__(self, key, value):

????????self.__dict__[key]?=?value

?

????def?__getattribute__(self, item):

????????#注意此處不要用 self.__dict__[item]

????????#因?yàn)閟elf.__dict__依然會被__getattribute__攔截 這樣就會陷入循環(huán)

????????return?object.__getattribute__(self,item)

?

a?=?Person('p1',?20)

print(a.hh)????????#這里會打印 don't have the attribute hh 以及 False

a.hh?=?'fdf'????????#這里設(shè)置該屬性值為'fdf'

print(a.hh)???????????#這里將打印出 fdf

  __delattr__ 定義當(dāng)一個屬性被刪除的時候的行為

  __get__ ?定義當(dāng)描述符的值被取得的時候的行為

  __set__ 定義當(dāng)描述符的值被設(shè)置的時候的行為

  __delete__ 定義當(dāng)描述符的值被刪除的時候的行為

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

class?Descriptor(object):

????def?__init__(self):

????????self.des?=?None

?

????def?__get__(self, instance, owner):

????????return?instance.__dict__.get(self.des,None)

?

????def?__set__(self, instance, value):

????????instance.__dict__[self.des]?=?value

?

?

class?Person(object):

????des?=?Descriptor()?#這里的Descriptor就是一個描述符類

?

????def?__init__(self, name, age):

????????self.name?=?name

????????self.age?=?age

????????self._registry?=?{

????????????'name': name,

????????????'age': age

????????}

?

?

a?=?Person('p1',?20)

a.des?=?10????#這里會調(diào)用Descriptor的__set__方法

print(a.des)???#這里會調(diào)用Descriptor的__get__方法

  關(guān)于描述符有疑惑的朋友可以自己百度也可參考這篇文章 http://www.geekfan.net/7862/ ,網(wǎng)上也有很多相關(guān)的文章,這里的重點(diǎn)不是描述符所以這里不會有關(guān)于描述符過多的解釋!

總結(jié)

以上是生活随笔為你收集整理的Python中的魔法方法的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。