溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Python上下文管理器Content Manager的示例分析

發(fā)布時(shí)間:2021-06-25 16:01:00 來源:億速云 閱讀:141 作者:小新 欄目:開發(fā)技術(shù)

這篇文章主要介紹了Python上下文管理器Content Manager的示例分析,具有一定借鑒價(jià)值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。

實(shí)踐

我們?cè)诖a實(shí)踐的時(shí)候,忽略了在同一代碼片段中,先打開文件,然后直接對(duì)文件進(jìn)行其他處理,因?yàn)檫@樣沒有任何意義,資源是處于被占用的情況。

先看下面檢測(cè)的代碼:

#!/usr/bin/env python
# _*_ coding: UTF-8 _*_
# MedusaSorcerer Script
import os


class OpenFile:
    def __init__(self):
        self.file = None

    def open(self, path):
        self.file = open(path, 'w')


if __name__ == '__main__':
    file_path = 'medusa.md'
    file = OpenFile()
    file.open(file_path)
    os.remove(file_path)

代碼中我們把文件對(duì)象,進(jìn)行了實(shí)例屬性的方式引用,在此之后,我們使用 os 模塊進(jìn)行刪除被寫入的文件。執(zhí)行改代碼片段后,會(huì)出現(xiàn)以下內(nèi)容:

Traceback (most recent call last):
  File "medusa/main.py", line 19, in <module>
    os.remove(file_path)
PermissionError: [WinError 32] 另一個(gè)程序正在使用此文件,進(jìn)程無法訪問。: 'medusa.md'

Process finished with exit code 1

那是因?yàn)楸粍h除的文件沒有得到資源釋放。我們?cè)谏厦娴幕A(chǔ)上進(jìn)行套用函數(shù)的方式:

#!/usr/bin/env python
# _*_ coding: UTF-8 _*_
# MedusaSorcerer Script
import os


class OpenFile:
    def __init__(self):
        self.file = None

    def open(self, path):
        self.file = open(path, 'w')


def open_file(path):
    file = OpenFile()
    file.open(path)


if __name__ == '__main__':
    file_path = 'medusa.md'
    open_file(file_path)
    os.remove(file_path)

這段代碼會(huì)成功的被執(zhí)行成功,原因是當(dāng)你執(zhí)行函數(shù)的時(shí)候,函數(shù)內(nèi)的臨時(shí)變量將被回收釋放,因此 OpenFile 的實(shí)例對(duì)象被釋放了,實(shí)例屬性也就不存在而被釋放,所以會(huì)執(zhí)行成功。

那是否我們的操作都應(yīng)該使用函數(shù)包裹的方式執(zhí)行呢?with 的出現(xiàn),完美解決了這個(gè)問題:

#!/usr/bin/env python
# _*_ coding: UTF-8 _*_
# MedusaSorcerer Script
import os

if __name__ == '__main__':
    file_path = 'medusa.md'
    with open(file_path, 'w') as f:
        print(f)
    os.remove(file_path)

在 with 語法中,將后面打開文件的操作,返回的文件對(duì)象,賦值給 f 變量,在結(jié)構(gòu)體中輸出了 f 變量的內(nèi)容,并且在結(jié)構(gòu)體外刪除了該文件:

medusa\python.exe medusa/main.py
<_io.TextIOWrapper name='medusa.md' mode='w' encoding='cp936'>

Process finished with exit code 0

在沒有使用 close() 的情況下,依舊可以對(duì)文件進(jìn)行刪除,這就是上下文管理的美妙。

實(shí)現(xiàn)

上下文管理,實(shí)際上是實(shí)現(xiàn)了 __enter__ 和 __exit__ 方法:

#!/usr/bin/env python
# _*_ coding: UTF-8 _*_
# MedusaSorcerer Script


class Medusa:

    def __init__(self):
        print('__init__')

    def __enter__(self):
        print('__enter__')

    def __exit__(self, exc_type, exc_val, exc_tb):
        print('__exit__')


if __name__ == '__main__':
    medusa = Medusa()
    with medusa:
        print('with object')
    print('finish')

以下是輸出結(jié)果:

__init__
__enter__
with object
__exit__
finish

我們發(fā)現(xiàn)魔法方法在結(jié)合某些語法后會(huì)發(fā)生自動(dòng)調(diào)度,所以,上下文管理中就在自動(dòng)調(diào)度中,關(guān)閉了某些對(duì)象。

優(yōu)點(diǎn)

實(shí)現(xiàn)上下文管理可以簡(jiǎn)化我們的代碼,讓代碼更加簡(jiǎn)單易讀,使用最少的代碼量,就可以完成全部工作。

感謝你能夠認(rèn)真閱讀完這篇文章,希望小編分享的“Python上下文管理器Content Manager的示例分析”這篇文章對(duì)大家有幫助,同時(shí)也希望大家多多支持億速云,關(guān)注億速云行業(yè)資訊頻道,更多相關(guān)知識(shí)等著你來學(xué)習(xí)!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI