栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 面试经验 > 面试问答

如何实现一个可以使用importlib即时修改源代码的导入挂钩?

面试问答 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

如何实现一个可以使用importlib即时修改源代码的导入挂钩?

find_module
并且
load_module
都已弃用。您需要分别切换到
find_spec
和(
create_module
exec_module
)模块。有关详细信息,请参见
importlib

文档。

您还需要检查是否要使用a

metaPathFinder
或a,
PathEntryFinder
因为调用它们的系统有所不同。也就是说,元路径查找器排在第一位,并且可以覆盖内置模块,而路径条目查找器专门用于在上找到的模块
sys.path

以下是一个非常基础的进口商,试图替换整个进口机器。它展示了如何使用功能(

find_spec
create_module
,和
exec_module
)。

import sysimport os.pathfrom importlib.abc import Loader, metaPathFinderfrom importlib.util import spec_from_file_locationclass MymetaFinder(metaPathFinder):    def find_spec(self, fullname, path, target=None):        if path is None or path == "": path = [os.getcwd()] # top level import --         if "." in fullname: *parents, name = fullname.split(".")        else: name = fullname        for entry in path: if os.path.isdir(os.path.join(entry, name)):     # this module has child modules     filename = os.path.join(entry, name, "__init__.py")     submodule_locations = [os.path.join(entry, name)] else:     filename = os.path.join(entry, name + ".py")     submodule_locations = None if not os.path.exists(filename):     continue return spec_from_file_location(fullname, filename, loader=MyLoader(filename),     submodule_search_locations=submodule_locations)        return None # we don't know how to import thisclass MyLoader(Loader):    def __init__(self, filename):        self.filename = filename    def create_module(self, spec):        return None # use default module creation semantics    def exec_module(self, module):        with open(self.filename) as f: data = f.read()        # manipulate data some way...        exec(data, vars(module))def install():    """Inserts the finder into the import machinery"""    sys.meta_path.insert(0, MymetaFinder())

接下来是一个稍微精致的版本,尝试重用更多的导入机制。这样,您只需要定义如何获取模块源即可。

import sysfrom os.path import isdirfrom importlib import invalidate_cachesfrom importlib.abc import SourceLoaderfrom importlib.machinery import FileFinderclass MyLoader(SourceLoader):    def __init__(self, fullname, path):        self.fullname = fullname        self.path = path    def get_filename(self, fullname):        return self.path    def get_data(self, filename):        """exec_module is already defined for us, we just have to provide a way        of getting the source pre of the module"""        with open(filename) as f: data = f.read()        # do something with data ...        # eg. ignore it... return "print('hello world')"        return dataloader_details = MyLoader, [".py"]def install():    # insert the path hook ahead of other path hooks    sys.path_hooks.insert(0, FileFinder.path_hook(loader_details))    # clear any loaders that might already be in use by the FileFinder    sys.path_importer_cache.clear()    invalidate_caches()


转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/626313.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号