仅使用 IPython 时出现 ImportError
ImportError when using IPython only
我有一个已编译的 .pyd 模块,它依赖于其他几个 DLL(Python mapscript 和 MapServer),我正尝试在 Windows 10 上的 IPython 中使用它。设置步骤如下:
- 创建了一个新的虚拟环境
- 已将 mapscript 安装到环境中
- 设置 PATH 以包含包含 MapServer DLL 的文件夹
- 当 运行 使用标准 venv Python
时,"import mapscript" 工作正常
- 启动 ipython 和 运行ning "import mapscript" 时出现以下错误:
ImportError: DLL load failed: The specified procedure could not be found.
到目前为止的调试步骤:
- sys.executable 在两种情况下都是相同的(venv Python 和 venv IPython)
- os.environ["PATH"] 相同
- sys.path in ipython 最后在 IPython 中有几个额外的路径 -
C:\Users\user\.ipython
和 'c:\virtualenvs\mapscript-jupyter3\lib\site-packages\IPython\extensions' 但在其他方面是相同的。删除这些路径不会更改错误。
- 我尝试使用进程监视器查找丢失的 DLL,但似乎都找到了。
新创建的 venvs 在 Python 2.7 和 Python 3.6 中都会出现同样的问题。
这个问题似乎与 ipython notebook can import a pyd module but the python interpter can't
相反
所以我的问题是 IPython 对 Python 环境做了什么会导致与标准 Python 的差异并导致 ImportError?
这花了很长时间才找到问题。使用 Process Monitor 很容易发现任何丢失的 DLL。然而,在这种情况下,找到了所有相关的 DLL,但其中一个没有我试图导入的 .pyd 文件(或关联的 DLL)使用的函数。
我设法将测试范围缩小到导致错误的 2 个命令 - 直接加载 PYD。
python -c "import _mapscript"
ipython -c "import _mapscript"
我尝试删除/修改 PATH,进出虚拟环境,以及 py2 和 py3 并且都产生了相同的错误。
然后我尝试了 运行 Process Monitor 并比较了结果——除了 ipython 为交互式 shell 加载了大量额外的 Python 库之外没有什么突出的。
Process Monitor 为每个事件包含一个方便的 Properties 选项,它还显示为 Process 加载了哪些模块。这些可以排序,然后复制到剪贴板。
我能够比较工作 Python 和损坏的 IPython 进程的输出。 IPython 包含来自根 Python 安装 (C:\Python36\DLLs
) 的几个额外的 .pyd 文件。我知道我试图加载的 pyd 使用了 sqlite,这是加载的模块之一(大概是因为 IPython 将所有输入命令存储在 sqlite 数据库中以便轻松访问历史记录)。暂时删除 _sqlite3.pyd 文件允许加载模块。
Python DLLs 文件夹优先于 PATH 上的文件夹,因此当前的修复是将 Python DLL 文件夹中的 sqlite3.dll 替换为 MapServer 使用的文件夹和一切正常。
我有一个已编译的 .pyd 模块,它依赖于其他几个 DLL(Python mapscript 和 MapServer),我正尝试在 Windows 10 上的 IPython 中使用它。设置步骤如下:
- 创建了一个新的虚拟环境
- 已将 mapscript 安装到环境中
- 设置 PATH 以包含包含 MapServer DLL 的文件夹
- 当 运行 使用标准 venv Python 时,"import mapscript" 工作正常
- 启动 ipython 和 运行ning "import mapscript" 时出现以下错误:
ImportError: DLL load failed: The specified procedure could not be found.
到目前为止的调试步骤:
- sys.executable 在两种情况下都是相同的(venv Python 和 venv IPython)
- os.environ["PATH"] 相同
- sys.path in ipython 最后在 IPython 中有几个额外的路径 -
C:\Users\user\.ipython
和 'c:\virtualenvs\mapscript-jupyter3\lib\site-packages\IPython\extensions' 但在其他方面是相同的。删除这些路径不会更改错误。 - 我尝试使用进程监视器查找丢失的 DLL,但似乎都找到了。
新创建的 venvs 在 Python 2.7 和 Python 3.6 中都会出现同样的问题。 这个问题似乎与 ipython notebook can import a pyd module but the python interpter can't
相反所以我的问题是 IPython 对 Python 环境做了什么会导致与标准 Python 的差异并导致 ImportError?
这花了很长时间才找到问题。使用 Process Monitor 很容易发现任何丢失的 DLL。然而,在这种情况下,找到了所有相关的 DLL,但其中一个没有我试图导入的 .pyd 文件(或关联的 DLL)使用的函数。
我设法将测试范围缩小到导致错误的 2 个命令 - 直接加载 PYD。
python -c "import _mapscript"
ipython -c "import _mapscript"
我尝试删除/修改 PATH,进出虚拟环境,以及 py2 和 py3 并且都产生了相同的错误。
然后我尝试了 运行 Process Monitor 并比较了结果——除了 ipython 为交互式 shell 加载了大量额外的 Python 库之外没有什么突出的。
Process Monitor 为每个事件包含一个方便的 Properties 选项,它还显示为 Process 加载了哪些模块。这些可以排序,然后复制到剪贴板。
我能够比较工作 Python 和损坏的 IPython 进程的输出。 IPython 包含来自根 Python 安装 (C:\Python36\DLLs
) 的几个额外的 .pyd 文件。我知道我试图加载的 pyd 使用了 sqlite,这是加载的模块之一(大概是因为 IPython 将所有输入命令存储在 sqlite 数据库中以便轻松访问历史记录)。暂时删除 _sqlite3.pyd 文件允许加载模块。
Python DLLs 文件夹优先于 PATH 上的文件夹,因此当前的修复是将 Python DLL 文件夹中的 sqlite3.dll 替换为 MapServer 使用的文件夹和一切正常。