《Python Cookbook 3rd》笔记(5.18):将文件描述符包装成文件对象
將文件描述符包裝成文件對象
問題
你有一個對應(yīng)于操作系統(tǒng)上一個已打開的 I/O 通道 (比如文件、管道、套接字等)的整型文件描述符,你想將它包裝成一個更高層的 Python 文件對象。
解法
一個文件描述符和一個打開的普通文件是不一樣的。文件描述符僅僅是一個由操作系統(tǒng)指定的整數(shù),用來指代某個系統(tǒng)的 I/O 通道。如果你碰巧有這么一個文件描述符,你可以通過使用 open() 函數(shù)來將其包裝為一個 Python 的文件對象。你僅僅只需要使用這個整數(shù)值的文件描述符作為第一個參數(shù)來代替文件名即可。例如:
# Open a low-level file descriptor import os fd = os.open('somefile.txt', os.O_WRONLY j os.O_CREAT)# Turn into a proper file f = open(fd, 'wt') f.write('hello world\n') f.close()當(dāng)高層的文件對象被關(guān)閉或者破壞的時候,底層的文件描述符也會被關(guān)閉。如果這個并不是你想要的結(jié)果,你可以給 open() 函數(shù)傳遞一個可選的 colsefd=False 。比如:
# Create a file object, but don't close underlying fd when done f = open(fd, 'wt', closefd=False) ...討論
在 Unix 系統(tǒng)中,這種包裝文件描述符的技術(shù)可以很方便的將一個類文件接口作用于一個以不同方式打開的 I/O 通道上,如管道、套接字等。舉例來講,下面是一個操作管道的例子:
from socket import socket, AF_INET, SOCK_STREAMdef echo_client(client_sock, addr):print('Got connection from', addr)# Make text-mode file wrappers for socket reading/writingclient_in = open(client_sock.fileno(), 'rt', encoding='latin-1',closefd=False)client_out = open(client_sock.fileno(), 'wt', encoding='latin-1',closefd=False)# Echo lines back to the client using file I/Ofor line in client_in:client_out.write(line)client_out.flush()client_sock.close()def echo_server(address):sock = socket(AF_INET, SOCK_STREAM)sock.bind(address)sock.listen(1)while True:client, addr = sock.accept()echo_client(client, addr)需要重點強(qiáng)調(diào)的一點是,上面的例子僅僅是為了演示內(nèi)置的 open() 函數(shù)的一個特性,并且也只適用于基于 Unix 的系統(tǒng)。如果你想將一個類文件接口作用在一個套接字并希望你的代碼可以跨平臺,請使用套接字對象的 makefile() 方法。但是如果不考慮可移植性的話,那上面的解決方案會比使用 makefile() 性能更好一點。
你也可以使用這種技術(shù)來構(gòu)造一個別名,允許以不同于第一次打開文件的方式使用它。例如,下面演示如何創(chuàng)建一個文件對象,它允許你輸出二進(jìn)制數(shù)據(jù)到標(biāo)準(zhǔn)輸出 (通常以文本模式打開):
import sys# Create a binary-mode file for stdout bstdout = open(sys.stdout.fileno(), 'wb', closefd=False) bstdout.write(b'Hello World\n') bstdout.flush()盡管可以將一個已存在的文件描述符包裝成一個正常的文件對象,但是要注意的是并不是所有的文件模式都被支持,并且某些類型的文件描述符可能會有副作用 (特別是涉及到錯誤處理、文件結(jié)尾條件等等的時候)。在不同的操作系統(tǒng)上這種行為也是不一樣,特別的,上面的例子都不能在非 Unix 系統(tǒng)上運(yùn)行。我說了這么多,意思就是讓你充分測試自己的實現(xiàn)代碼,確保它能按照期望工作。
總結(jié)
以上是生活随笔為你收集整理的《Python Cookbook 3rd》笔记(5.18):将文件描述符包装成文件对象的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 《Python Cookbook 3rd
- 下一篇: 推荐算法--推荐系统架构(06)