为什么re模块试图导入enum.IntFlag?

时间:2017-12-18 23:45:17

标签: python python-3.x pydev development-environment archlinux

核心模块如何从另一个核心模块导入不存在的名称?

具体而言,re模块导入enum.IntFlag

重现方式

可以通过启动解释器并尝试导入,运行依赖于enum.IntFlag的程序(例如pip,在Eclipse中查看解释器设置页面,使用文本编辑器,运行PyDev中的交互式控制台,......

直接导入

from enum import IntFlag

Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
ImportError: cannot import name 'IntFlag'

运行依赖程序(pip)

Traceback (most recent call last):
  File "/usr/bin/pip", line 4, in <module>
    import re
  File "/usr/lib/python3.6/re.py", line 142, in <module>
    class RegexFlag(enum.IntFlag):
AttributeError: module 'enum' has no attribute 'IntFlag'

在Eclipse中查看Python解释器设置也会产生上述回溯(在解释器设置所在的窗口中)。

Preferences =:> PyDev =:> Interpreters =:> Python Interpreter

此外,文本编辑器无法识别打印声明。

另外,当我运行控制台解释器(例如:Ctrl+Alt+Enter时,eclipse失败,弹出窗口显示:

  

'创建交互式控制台'遇到了问题。

     

初始化控制台时出错。

     

错误详情

单击错误详细信息按钮:

Error initializing console.
Unexpected error connecting to console.
Failed to recive suitable Hello response from pydevconsole. Last msg received: Console already exited with value: 1 while waiting for an answer.

Command Line used:  /usr/bin/python3.6 -u /usr/lib/eclipse/../../../home/scott/.eclipse/org.eclipse.platform_4.7.1_155965261_linux_gtk_x86_64/plugins/org.python.pydev_6.2.0.201711281614/pysrc/pydevconsole.py 44633 43575 44633 43575

Environment:
PATH=/usr/local/sbin:/usr/local/bin:/usr/bin:/opt/cuda/bin:/usr/lib/jvm/default/bin:/opt/jython/bin/:/usr/bin/site_perl:/usr/bin/vendor_perl:/usr/bin/core_perl
XAUTHORITY=/tmp/xauth-1000-_0
XDG_DATA_DIRS=/usr/share:/usr/share:/usr/local/share
DBUS_SESSION_BUS_ADDRESS=unix:path=/run/user/1000/bus
XDG_CURRENT_DESKTOP=KDE
QT_AUTO_SCREEN_SCALE_FACTOR=0
MAIL=/var/spool/mail/scott
MOZ_PLUGIN_PATH=/usr/lib/mozilla/plugins
QT_LINUX_ACCESSIBILITY_ALWAYS_ON=1
SESSION_MANAGER=local/scott-pc:@/tmp/.ICE-unix/976,unix/scott-pc:/tmp/.ICE-unix/976
LOGNAME=scott
PAM_KWALLET5_LOGIN=/run/user/1000/kwallet5.socket
PWD=/home/scott
XCURSOR_THEME=Azenis
PYPATH=/code/scott/Py/Path
PYTHONPATH=/home/scott/.eclipse/org.eclipse.platform_4.7.1_155965261_linux_gtk_x86_64/plugins/org.python.pydev_6.2.0.201711281614/pysrc/pydev_sitecustomize:/mnt/ssdata/scott/code/Py/Path:/home/scott:/usr/lib/python3.6/Tools/scripts:/usr/lib/python3.6/lib-dynload:/usr/lib/python3.6/site-packages:/usr/lib/python3.6
KDE_SESSION_VERSION=5
SHELL=/bin/bash
KDE_MULTIHEAD=false
KDE_FULL_SESSION=true
GTK_MODULES=canberra-gtk-module
GDK_SCALE=1
DATA=/data/scott/
XDG_SESSION_PATH=/org/freedesktop/DisplayManager/Session1
PYDYNLOAD=/usr/lib/python3.6/lib-dynload
VM=/vm/scott/
XDG_SESSION_DESKTOP=KDE
SHLVL=1
PYSITE=/usr/lib/python3.6/site-packages
OXYGEN_DISABLE_INNER_SHADOWS_HACK=1
PYVER=3.6
KDE_SESSION_UID=1000
XFILESEARCHPATH=/usr/dt/app-defaults/%L/Dt
LANG=en_GB.UTF-8
XDG_SEAT_PATH=/org/freedesktop/DisplayManager/Seat0
XDG_SESSION_ID=c2
XDG_SESSION_TYPE=x11
DISPLAY=:0
PYTHONSTARTUP=/usr/lib/python3.6/startup_hook.py
GTK_RC_FILES=/etc/gtk/gtkrc:/home/scott/.gtkrc:/home/scott/.config/gtkrc
ECLIPSE_HOME=/usr/lib/eclipse
XDG_SESSION_CLASS=user
XCURSOR_SIZE=56
LIBOVERLAY_SCROLLBAR=0
DESKTOP_SESSION=/usr/share/xsessions/plasma
GDK_CORE_DEVICE_EVENTS=1
USER=scott
GTK2_RC_FILES=/etc/gtk-2.0/gtkrc:/home/scott/.gtkrc-2.0:/home/scott/.config/gtkrc-2.0
XDG_SEAT=seat0
CODE=/code/scott/
GS_LIB=/home/scott/.fonts
PYSCRIPTS=/usr/lib/python3.6/Tools/scripts
NLSPATH=/usr/dt/lib/nls/msg/%L/%N.cat
XDG_VTNR=1
XDG_RUNTIME_DIR=/run/user/1000
HOME=/home/scott
PYTHONIOENCODING=UTF-8
PYDEV_UMD_ENABLED=true
PYDEV_UMD_NAMELIST=guidata,guiqwt
PYDEV_UMD_VERBOSE=true

进度

在标准库版本/usr/lib/python3.6/enum.py中, 名称enum.IntFlag存在于enum.__all__

是否从标准库版本导入enum

import enum
print(enum.__file__)
# /usr/lib/python3.6/site-packages/enum/__init__.py

/usr/lib/python3.6/site-packages/enum/__init__.py导入我检查了该文件,但没有IntFlag。此外,它是一个包init而不是一个文件。

没有/usr/lib/python3.6/site-packages/enum/enum.py。有一个README文件说明这是3.4版。

我可以用__init__.py

替换enum.py的内容

我可以删除site-packages/enum目录

解决方案

还有另一个enum模块,一个python3.4兼容包,它掩盖了标准库中的模块版本。

python34兼容包:/usr/lib/python3.6/enum/__init__.py

标准库模块:/usr/lib/python3.6/enum.py

我删除了目录:

mkdir enumbackup
cd enumbackup
sudo mv /usr/lib/python3.6/enum .
sudo mv /usr/lib/python3.6/enumenum34-1.1.6.dist-info .

现在,enum.__file__会显示/usr/lib/python3.6/enum.py,但一切正常。

from enum import IntFlag
## works

感谢您的帮助! :)

1 个答案:

答案 0 :(得分:5)

您应该检查您的python路径上是否有另一个名为enum的模块正在影响stdlib enum模块。要做到这一点,你可以这样做:

import enum
print(enum.__file__)

如果这与python3 stdlib模块路径不匹配,则尝试删除它。理想情况下,您应该使用系统使用的任何软件包管理器来卸载它,因为它可能是某些其他软件包的依赖项。但如果不可能,您可以尝试手动删除它。