找到dll,但无法从当前工作目录中加载

时间:2019-10-12 18:11:12

标签: python dll pyinstaller ctypes

我正在运行以下代码:

def PATH():
    return "\n"+"\n".join(os.environ["PATH"].split(os.pathsep))

def syspath():
    return "\n"+"\n".join(sys.path)

def test_haha():
    logger.info(f"CWD: {os.getcwd()}")
    logger.info(f"sys.path: {syspath()}")
    logger.info(f"PATH: {PATH()}")

    logger.info(f"Found? {str(util.find_library('PCANBasic'))}")
    windll.LoadLibrary("PCANBasic")
    # pcanbasic_path = str(Path(os.getcwd()) / "PCANBasic.dll")

我的PCANBasic.dll坐在我的工作目录中。我正在使用Windows 10。

如果我从命令行运行python,则一切正常。

但是,我有一个用pyqt编写并与PyInstaller捆绑在一起的IDE。该IDE具有运行python文件的功能(在运行与我手动运行的命令相同的批处理文件中调用QProcess),在这种情况下,它将失败,并显示以下内容:

    def test_haha():
        logger.info(f"CWD: {os.getcwd()}")
        logger.info(f"sys.path: {syspath()}")
        logger.info(f"PATH: {PATH()}")

        logger.info(f"Found? {str(util.find_library('PCANBasic'))}")
>       windll.LoadLibrary("PCANBasic")

demo_test\test_dll.py:23: 
_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _
C:\Users\myuser\python3_portable\lib\ctypes\__init__.py:434: in LoadLibrary
    return self._dlltype(name)
_ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _

self = <WinDLL 'PCANBasic', handle 0 at 0x1e9e6480668>, name = 'PCANBasic'
mode = 0, handle = None, use_errno = False, use_last_error = False

    def __init__(self, name, mode=DEFAULT_MODE, handle=None,
                 use_errno=False,
                 use_last_error=False):
self._name = name
        flags = self._func_flags_
        if use_errno:
            flags |= _FUNCFLAG_USE_ERRNO
        if use_last_error:
            flags |= _FUNCFLAG_USE_LASTERROR
        if _sys.platform.startswith("aix"):
            """When the name contains ".a(" and ends with ")",
               e.g., "libFOO.a(libFOO.so)" - this is taken to be an
               archive(member) syntax for dlopen(), and the mode is adjusted.
               Otherwise, name is presented to dlopen() as a file argument.
            """
            if name and name.endswith(")") and ".a(" in name:
                mode |= ( _os.RTLD_MEMBER | _os.RTLD_NOW )

        class _FuncPtr(_CFuncPtr):
            _flags_ = flags
            _restype_ = self._func_restype_
        self._FuncPtr = _FuncPtr

        if handle is None:
>           self._handle = _dlopen(self._name, mode)
E           OSError: [WinError 126] The specified module could not be found

cwd sys.path相同,PATH几乎相同(IDE版本具有PyQt5 bin文件夹)。

总而言之,这些问题是:

1-是什么原因导致这种行为差异,在一种情况下,我能够从当前工作目录中加载dll,而在另一种情况下,则无法?

2-在任何情况下,如果dll位于PATH中的任何目录中,它将起作用。依靠工作目录中的dll并将其正确安装在某个位置,可能不是一个好习惯吗?

谢谢!

1 个答案:

答案 0 :(得分:0)

我能够使用Process Monitor并按照以下说明进行过滤:

  • 操作是CreateFile
  • 路径包含PCANBasic

我能够看到PyInstaller捆绑包的搜索路径不同于直接命令。

然后我发现this issue提到PyInstaller使用SetDllDirectoryA调用将捆绑包自己的文件夹添加到DLL搜索路径,但是具有删除当前目录的副作用。

相关问题