我的情况是,同一个 Python 模块存在于两个不同版本的同一个目录中; mymodule.py
和 mymodule.so
(我通过 Cython 从第一个中获得后者,但这与我的问题无关)。当我使用 Python 时
import mymodule
它总是选择 mymodule.so
。有时我真的很想导入 mymodule.py
。我可以暂时将 mymodule.so
移动到另一个位置,但是如果我同时运行另一个需要导入 mymodule.so
的 Python 实例,那效果就不好了。
问题是如何使 import
更喜欢 .py
文件而不是 .so
,而不是相反?
以下是我对解决方案的看法:
我想象使用 importlib
表演一些魔术并可能编辑 sys.meta_path
。具体来说,我看到 sys.meta_path[2]
包含 _frozen_importlib_external.PathFinder
用于导入外部模块,即这用于 mymodule.py
和 mymodule.so
。如果我可以用类似的 PathFinder
替换它,它对文件类型使用反向排序,我就会有一个解决方案。
如果这会影响解决方案,我正在使用 Python 3.7。
编辑
请注意,简单地阅读 mymodule.py
的源代码行并执行
它们是不行的,因为 mymodule.py
可能本身导入其他模块,这些模块再次存在于 .py
和 .so
版本中(我想导入这些的 .py
版本作为好吧)。
最佳答案
这是另一种解决方案,只需调整运行时默认生成的查找器即可。这使用了隐藏的实现细节 (FileFinder._loaders
),但我已经在 CPython 3.7、3.8 和 3.9 上进行了测试。
from contextlib import contextmanager
from dataclasses import dataclass
from importlib.machinery import FileFinder
from importlib.abc import Finder
import sys
from typing import Callable
@dataclass
class PreferPureLoaderHook:
orig_hook: Callable[[str], Finder]
def __call__(self, path: str) -> Finder:
finder = self.orig_hook(path)
if isinstance(finder, FileFinder):
# Move pure python file loaders to the front
finder._loaders.sort(key=lambda pair: 0 if pair[0] in (".py", ".pyc") else 1) # type: ignore
return finder
@contextmanager
def prefer_pure_python_imports():
sys.path_hooks = [PreferPureLoaderHook(h) for h in sys.path_hooks]
sys.path_importer_cache.clear()
yield
assert all(isinstance(h, PreferPureLoaderHook) for h in sys.path_hooks)
sys.path_hooks = [h.orig_hook for h in sys.path_hooks]
sys.path_importer_cache.clear()
with prefer_pure_python_imports():
...
关于Python:强制导入更喜欢 .py 而不是 .so,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56074754/