成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Python中的魔法方法

開發 前端
python中的魔法方法是一些可以讓你對類添加“魔法”的特殊方法,它們經常是兩個下劃線包圍來命名的

python中的魔法方法是一些可以讓你對類添加“魔法”的特殊方法,它們經常是兩個下劃線包圍來命名的。

Python的魔法方法,也稱為dunder(雙下劃線)方法。大多數的時候,我們將它們用于簡單的事情,例如構造函數(init)、字符串表示(str, repr)或算術運算符(add/mul)。其實還有許多你可能沒有聽說過的但是卻很好用的方法,在這篇文章中,我們將整理這些魔法方法!

迭代器的大小

我們都知道__len__方法,可以用它在容器類上實現len()函數。但是,如果您想獲取實現迭代器的類對象的長度怎么辦?

it = iter(range(100))
print(it.__length_hint__())
# 100
next(it)
print(it.__length_hint__())
# 99

a = [1, 2, 3, 4, 5]
it = iter(a)
print(it.__length_hint__())
# 5
next(it)
print(it.__length_hint__())
# 4
a.append(6)
print(it.__length_hint__())
# 5

你所需要做的就是實現__length_hint__方法,這個方法是迭代器上的內置方法(不是生成器),正如你上面看到的那樣,并且還支持動態長度更改。但是,正如他的名字那樣,這只是一個提示(hint),并不能保證完全準確:對于列表迭代器,可以得到準確的結果,但是對于其他迭代器則不確定。但是即使它不準確,它也可以幫我們獲得需要的信息,正如PEP 424中解釋的那樣。

length_hint must return an integer (else a TypeError is raised) or NotImplemented, and is not required to be accurate.  It may return a value that is either larger or smaller than the actual size of the container.  A return value of NotImplemented indicates that there is no finite length estimate.  It may not return a negative value (else a ValueError is raised).

元編程

大部分很少看到的神奇方法都與元編程有關,雖然元編程可能不是我們每天都需要使用的東西,但有一些方便的技巧可以使用它。

一個這樣的技巧是使用__init_subclass__作為擴展基類功能的快捷方式,而不必處理元類:

 class Pet:
def __init_subclass__(cls, /, default_breed, **kwargs):
super().__init_subclass__(**kwargs)
cls.default_breed = default_breed

class Dog(Pet, default_name="German Shepherd"):
pass

上面的代碼我們向基類添加關鍵字參數,該參數可以在定義子類時設置。在實際用例中可能會在想要處理提供的參數而不僅僅是賦值給屬性的情況下使用此方法。

看起來非?;逎⑶液苌贂玫?,但其實你可能已經遇到過很多次了,因為它一般都是在構建API時使用的,例如在SQLAlchemy或Flask Views中都使用到了。

另一個元類的神奇方法是__call__。這個方法允許自定義調用類實例時發生的事情:

 class CallableClass:
def __call__(self, *args, **kwargs):
print("I was called!")

instance = CallableClass()

instance()
# I was called!

可以用它來創建一個不能被調用的類:

 class NoInstances(type):
def __call__(cls, *args, **kwargs):
raise TypeError("Can't create instance of this class")

class SomeClass(metaclass=NoInstances):
@staticmethod
def func(x):
print('A static method')

instance = SomeClass()
# TypeError: Can't create instance of this class

對于只有靜態方法的類,不需要創建類的實例就用到了這個方法。

另一個類似的場景是單例模式——一個類最多只能有一個實例:

 class Singleton(type):
def __init__(cls, *args, **kwargs):
cls.__instance = None
super().__init__(*args, **kwargs)

def __call__(cls, *args, **kwargs):
if cls.__instance is None:
cls.__instance = super().__call__(*args, **kwargs)
return cls.__instance
else:
return cls.__instance

class Logger(metaclass=Singleton):
def __init__(self):
print("Creating global Logger instance")

Singleton類擁有一個私有__instance——如果沒有,它會被創建并賦值,如果它已經存在,它只會被返回。

假設有一個類,你想創建它的一個實例而不調用__init__。__new__ 方法可以幫助解決這個問題:

 class Document:
def __init__(self, text):
self.text = text

bare_document = Document.__new__(Document)
print(bare_document.text)
# AttributeError: 'Document' object has no attribute 'text'

setattr(bare_document, "text", "Text of the document")

在某些情況下,我們可能需要繞過創建實例的通常過程,上面的代碼演示了如何做到這一點。我們不調用Document(…),而是調用Document.__new__(Document),它創建一個裸實例,而不調用__init__。因此,實例的屬性(在本例中為text)沒有初始化,所欲我們需要額外使用setattr函數賦值(它也是一個魔法的方法__setattr__)。

為什么要這么做呢。因為我們可能會想要替代構造函數,比如:

 class Document:
def __init__(self, text):
self.text = text

@classmethod
def from_file(cls, file): # Alternative constructor
d = cls.__new__(cls)
# Do stuff...
return d

這里定義from_file方法,它作為構造函數,首先使用__new__創建實例,然后在不調用__init__的情況下配置它。

下一個與元編程相關的神奇方法是__getattr__。當普通屬性訪問失敗時調用此方法。這可以用來將對缺失方法的訪問/調用委托給另一個類:

 class String:
def __init__(self, value):
self._value = str(value)

def custom_operation(self):
pass

def __getattr__(self, name):
return getattr(self._value, name)

s = String("some text")
s.custom_operation() # Calls String.custom_operation()
print(s.split()) # Calls String.__getattr__("split") and delegates to str.split
# ['some', 'text']

print("some text" + "more text")
# ... works
print(s + "more text")
# TypeError: unsupported operand type(s) for +: 'String' and 'str'

我們想為類添加一些額外的函數(如上面的custom_operation)定義string的自定義實現。但是我們并不想重新實現每一個字符串方法,比如split、join、capitalize等等。這里我們就可以使用__getattr__來調用這些現有的字符串方法。

雖然這適用于普通方法,但請注意,在上面的示例中,魔法方法__add__(提供的連接等操作)沒有得到委托。所以,如果我們想讓它們也能正常工作,就必須重新實現它們。

自省(introspection)

最后一個與元編程相關的方法是__getattribute__。它一個看起來非常類似于前面的__getattr__,但是他們有一個細微的區別,__getattr__只在屬性查找失敗時被調用,而__getattribute__是在嘗試屬性查找之前被調用。

所以可以使用__getattribute__來控制對屬性的訪問,或者你可以創建一個裝飾器來記錄每次訪問實例屬性的嘗試:

 def logger(cls):
original_getattribute = cls.__getattribute__

def getattribute(self, name):
print(f"Getting: '{name}'")
return original_getattribute(self, name)

cls.__getattribute__ = getattribute
return cls

@logger
class SomeClass:
def __init__(self, attr):
self.attr = attr

def func(self):
...

instance = SomeClass("value")
instance.attr
# Getting: 'attr'
instance.func()
# Getting: 'func'

裝飾器函數logger 首先記錄它所裝飾的類的原始__getattribute__方法。然后將其替換為自定義方法,該方法在調用原始的__getattribute__方法之前記錄了被訪問屬性的名稱。

魔法屬性

到目前為止,我們只討論了魔法方法,但在Python中也有相當多的魔法變量/屬性。其中一個是__all__:

 # some_module/__init__.py
__all__ = ["func", "some_var"]

some_var = "data"
some_other_var = "more data"

def func():
return "hello"

# -----------

from some_module import *

print(some_var)
# "data"
print(func())
# "hello"

print(some_other_var)
# Exception, "some_other_var" is not exported by the module

這個屬性可用于定義從模塊導出哪些變量和函數。我們創建了一個Python模塊…/some_module/單獨文件(__init__.py)。在這個文件中定義了2個變量和一個函數,只導出其中的2個(func和some_var)。如果我們嘗試在其他Python程序中導入some_module的內容,我們只能得到2個內容。

但是要注意,__all__變量只影響上面所示的* import,我們仍然可以使用顯式的名稱導入函數和變量,比如import some_other_var from some_module。

另一個常見的雙下劃線變量(模塊屬性)是__file__。這個變量標識了訪問它的文件的路徑:

 from pathlib import Path

print(__file__)
print(Path(__file__).resolve())
# /home/.../directory/examples.py

# Or the old way:
import os
print(os.path.dirname(os.path.abspath(__file__)))
# /home/.../directory/

這樣我們就可以結合__all__和__file__,可以在一個文件夾中加載所有模塊:

# Directory structure:
# .
# |____some_dir
# |____module_three.py
# |____module_two.py
# |____module_one.py

from pathlib import Path, PurePath
modules = list(Path(__file__).parent.glob("*.py"))
print([PurePath(f).stem for f in modules if f.is_file() and not f.name == "__init__.py"])
# ['module_one', 'module_two', 'module_three']

最后一個我重要的屬性是的是__debug__。它可以用于調試,但更具體地說,它可以用于更好地控制斷言:

 # example.py
def func():
if __debug__:
print("debugging logs")

# Do stuff...

func()

如果我們使用python example.py正常運行這段代碼,我們將看到打印出“調試日志”,但是如果我們使用python -O example.py,優化標志(-O)將把__debug__設置為false并刪除調試消息。因此,如果在生產環境中使用-O運行代碼,就不必擔心調試過程中被遺忘的打印調用,因為它們都不會顯示。

創建自己魔法方法?

我們可以創建自己的方法和屬性嗎?是的,你可以,但你不應該這么做。

雙下劃線名稱是為Python語言的未來擴展保留的,不應該用于自己的代碼。如果你決定在你的代碼中使用這樣的名稱,那么將來如果它們被添加到Python解釋器中,這就與你的代碼不兼容了。所以對于這些方法,我們只要記住和使用就好了。

責任編輯:華軒 來源: DeepHub IMBA
相關推薦

2022-07-07 08:01:51

Python魔法方法代碼

2021-01-15 07:55:09

Python腳本語言

2024-08-19 10:21:37

接口Python魔法方法

2023-11-09 08:55:17

Python雙下劃線

2024-10-29 10:54:07

2010-02-05 11:23:01

C++聲明語法

2020-04-03 13:43:23

Python列表推導式字典推導式

2023-12-08 07:52:51

Spring項目開發

2021-11-29 07:02:24

Python函數操作

2022-05-20 12:40:23

PythonMetaclass

2022-07-28 08:33:59

JupyterPython

2023-10-19 07:42:37

Python數據類型

2023-07-04 08:33:46

Python對象編程

2021-07-16 10:23:47

Python設計對象

2021-03-04 07:24:32

開發者搜索編程

2024-02-21 20:43:02

Python列表推導式

2017-02-05 10:06:53

Python黑魔法描述符

2016-10-19 15:15:26

2023-10-26 08:28:31

Python數據去重

2024-04-11 08:58:59

Python構造函數對象編程
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产亚洲精品美女久久久久久久久久 | 欧美激情综合色综合啪啪五月 | 免费人成在线观看网站 | 欧美日韩亚洲视频 | 日韩一区二区精品 | 久久中文字幕一区 | 国产精品久久久久久久久久软件 | 狠狠色综合欧美激情 | 中日av| 久久之精品 | 久久亚洲综合 | 国产在线视频一区 | 日日干干 | 亚洲免费视频网址 | 精品粉嫩aⅴ一区二区三区四区 | 日本精品视频一区二区 | 免费黄色在线观看 | 99免费精品视频 | 欧美激情精品久久久久久变态 | 国产精品久久久久久婷婷天堂 | 国产免费视频 | 伊人春色在线观看 | 香蕉视频久久久 | 不卡在线视频 | 亚洲成人中文字幕 | 色综合99 | 国产成人免费视频网站视频社区 | 天堂色 | 日本精品在线观看 | 日韩成人专区 | 日韩欧美久久精品 | 在线免费看黄 | 日韩精品在线网站 | 男人久久天堂 | 精品国产1区2区3区 一区二区手机在线 | 日日摸日日添日日躁av | 伊人看片| 毛色毛片免费看 | 九九久久精品 | 中文字幕日韩一区 | 日韩欧美一区二区三区 |