在Python 2.7中,我得到以下结果:
>>> with open("README.md", "r") as fin:
... print(isinstance(fin, file))
...
True
在python 3.5中我得到:
>>> with open("README.md", "r") as fin:
... print(isinstance(fin, file))
...
Traceback (most recent call last):
File "<stdin>", line 2, in <module>
NameError: name 'file' is not defined
所以,好吧,我查看了Python文档,发现在Python 3.5中,文件的类型是io.IOBase
(或某个子类)。让我来看看:
>>> import io
>>> with open("README.md", "r") as fin:
... print(isinstance(fin, io.IOBase))
...
True
但当我尝试Python 2.7:时
>>> import io
>>> with open("README.md", "r") as fin:
... print(isinstance(fin, io.IOBase))
...
False
所以在这一点上,我很困惑。查看文档,我觉得Python 2.7应该报告True
。
很明显,我错过了一些基本的东西,也许是因为现在是美国东部时间下午6:30,但我有两个相关的问题:
- 为什么Python为
isinstance(fin, io.IOBase)
报告False
- 有没有一种方法可以测试变量是否是一个在Python 2.7和3.5中都能工作的打开文件
来自链接文档:
在Python2.x下,这被提议作为内置文件对象的替代方案
因此,它们在python2.x中并不相同。
至于第二部分,这适用于蟒蛇2和3,尽管不是世界上最漂亮的东西:
import io
try:
file_types = (file, io.IOBase)
except NameError:
file_types = (io.IOBase,)
with open("README.md", "r") as fin:
print(isinstance(fin, file_types))
对于python2
import types
f = open('test.txt', 'r') # assuming this file exists
print (isinstance(f,types.FileType))
对于蟒蛇3
import io
import types
f1 = open('test.txt', 'r') # assuming this file exists
f2 = open('test.txt', 'rb') # assuming this file exists
print (isinstance(f1,io.IOBase))
print (isinstance(f2,io.IOBase))
(编辑:我以前的解决方案针对io.TextIOWrapper进行了测试,它只适用于以文本模式打开的文件。请参阅https://docs.python.org/3/library/io.html#class-描述python3类层次结构的层次结构)。