如何在 python 2 和 3 中使用 unicode_literals 获得兼容的 type() 行为?

mpo*_*ett 3 python python-2.7 python-unicode python-3.6

这个问题看起来与这个问题非常相似,但是评论中的建议不起作用(不再?),如下所示。

我正在尝试编写一个 python2-3 兼容包,我的方法之一有一个类生成器,并且type()在 python-2.7 测试中给我带来了问题:

Python 2.7.13 (default, Mar 18 2017, 17:03:32) 
[GCC 4.2.1 Compatible Apple LLVM 7.0.2 (clang-700.1.81)] on darwin
Type "help", "copyright", "credits" or "license" for more information.
>>> from __future__ import unicode_literals
>>> from builtins import str
>>> type('MyClass', (object,), {})
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: type() argument 1 must be string, not unicode
>>> type(str('MyClass'), (object,), {})
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: type() argument 1 must be string, not newstr
Run Code Online (Sandbox Code Playgroud)

Python -Future 概述页面说:

# Compatible output from isinstance() across Py2/3:
assert isinstance(2**64, int)        # long integers
assert isinstance(u'blah', str)
assert isinstance('blah', str)       # only if unicode_literals is in effect
Run Code Online (Sandbox Code Playgroud)

我预计这会给我在任何需要字符串的地方提供一致的行为,但显然不是。

正确的、与版本无关的方法是什么?我链接到的另一个问题是在 python-2.6 时代提出的,从那时起,行为似乎已经发生了变化。我不认为我可以直接转储 unicode_literals,因为如果我没有的话,我会在调用 hashlib 时遇到可移植性问题(在其他地方)。

Mar*_*ers 5

不要使用,使用Python 版本附带的builtins.str()普通格式:str

>>> from __future__ import unicode_literals
>>> type(str('MyClass'), (object,), {})
<class '__main__.MyClass'>
Run Code Online (Sandbox Code Playgroud)

这在 Python 2 和 3 中都适用。如果默认情况下该future.builtins模块替换str内置类型,请使用该__builtin__模块:

try:
    # Python 2
    from __builtin__ import str as builtin_str
except ImportError:
    # Python 3
    from builtins import str as builtin_str

MyClass = type(builtin_str('MyClass'), (object,), {})
Run Code Online (Sandbox Code Playgroud)