一、fileno简介
fileno是Python中一个比较基础的函数,用于返回文件描述符。文件描述符是在内核中用于标识一个特定的文件的整数值,通常与UNIX系统相关联。在UNIX系统中,每个进程都有一个文件描述符表,用于管理文件、网络套接字和其他输入/输出资源。使用fileno函数可以获取文件描述符,进而进行文件读写等操作。
二、fileno的用法
在Python中,fileno函数可以应用于三种类型的对象:文件、套接字和管道。下面分别介绍一下三种情况下如何使用fileno函数。
1.文件操作
在文件操作中,我们通常需要先打开一个文件对象,然后才能进行文件读写等操作。fileno函数可用于获取文件对象的文件描述符。其代码示例如下:
f = open('example.txt', 'r')
fd = f.fileno()
print(fd)
代码执行结果将返回文件对象的文件描述符。
2.网络套接字
在网络编程中,我们经常需要使用套接字进行数据传输。fileno函数也可用于获取套接字的文件描述符。其代码示例如下:
import socket
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.bind(('localhost', 8888))
s.listen(1)
fd = s.fileno()
print(fd)
代码执行结果将返回套接字对象的文件描述符。
3.管道
在进程间通信中,我们可以使用管道(pipe)进行数据传输。fileno函数也可用于获取管道的文件描述符。其代码示例如下:
import os r, w = os.pipe() reader_fd = os.fdopen(r) writer_fd = os.fdopen(w) print(reader_fd.fileno(), writer_fd.fileno())
代码执行结果将返回管道的两个文件描述符。
三、fileno函数的进阶用法:简单到匿名
除了上述三种基本情况,fileno函数还有一些进阶用法。本部分将介绍fileno的另外三种情况:匿名管道、匿名和非阻塞I/O。
1.匿名管道
在UNIX系统中,管道(pipe)是一种可用于协调进程间通信的IPC机制。UNIX管道分为有名管道和匿名管道,其中匿名管道不指向文件系统中的任何文件,而是只在进程间传递数据。在Python中,我们可以使用os模块中的pipe函数创建一个匿名管道,并使用fileno函数获取其文件描述符,实现进程间通信。其代码示例如下:
import os
r, w = os.pipe()
pid = os.fork()
if pid:
os.close(w)
r = os.fdopen(r)
print('Parent got:', r.readline())
else:
os.close(r)
w = os.fdopen(w, 'w')
print('child sending')
w.write('hello parent\n')
w.close()
代码执行结果将在父进程打印”Parent got: hello parent”,在子进程打印”child sending”。
2.匿名
我们可以利用fileno函数以及ctypes模块中的memmove函数,来实现Python中的匿名内存映射。其代码示例如下:
import mmap
import ctypes
import os
m = mmap.mmap(-1,4096, mmap.MAP_ANONYMOUS | mmap.MAP_PRIVATE)
m.write(b'hello')
os.dup2(ctypes.c_int(m.fileno()).value,0)
os.system('cat')
代码执行结果将在命令行打印”hello”。
3.非阻塞I/O
在进行I/O操作时,如果程序一直等待I/O操作完成之后才能继续执行,就会产生阻塞。为了避免阻塞,并且不影响程序的执行,可以使用非阻塞的I/O。Python的select函数是用于多路复用的一个函数,我们可以使用fileno函数将文件描述符与select函数结合起来,实现非阻塞的I/O操作。其代码示例如下:
import select
import socket
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.bind(('localhost', 8888))
s.listen(1)
fd = s.fileno()
rlist = [s]
wlist = []
while True:
readable, writable, exceptional = select.select(rlist, wlist, rlist)
for s in readable:
if s is r:
conn, addr = s.accept()
rlist.append(conn)
else:
data = s.recv(1024)
if data == b'':
rlist.remove(s)
else:
wlist.append(s)
for s in writable:
s.send(b'hello')
wlist.remove(s)
代码执行结果将在控制台输出”hello”。
原创文章,作者:BICZ,如若转载,请注明出处:https://www.506064.com/n/138128.html
微信扫一扫
支付宝扫一扫