一、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"。