导入路径 - 正确的方法?

2024-01-10

我知道有很多类似或相同的问题,但我仍然无法理解/找到使用模块的正确方法。 Python 是我最喜欢的语言,除了使用导入之外,我喜欢其中的所有内容:递归导入(当您尝试引用尚不存在的名称时)、导入路径等。

所以,我有这样的项目结构:

my_project/
    package1/
        __init__.py
        module1
        module2
    package2/
        __init__.py
        module1
        module2

Package1可以作为独立装置使用,但也有望通过进口package2。 我现在在做什么,例如,在package1.module1我写的from package1 import module2,即使用导入模块的完整路径。我这样做是因为如果我使用import module2-- 当模块从另一个包导入时,这将不起作用(package2)。我也无法使用from . import module2-- 这在运行时不起作用module1直接地。

好的,所以对于from package1 import module2 in package1.module1在两种情况下都可以工作(直接运行时package1.module1当从以下位置导入它时package2)我在开头添加这些行package1.module1:

import os, sys
currDir = os.path.dirname(os.path.realpath(__file__))
rootDir = os.path.abspath(os.path.join(currDir, '..'))
if rootDir not in sys.path: # add parent dir to paths
    sys.path.append(rootDir)

对我来说这可行,但我觉得这不是Pythonic。难道我做错了什么?

相反,我应该总是跑步吗package1.module1从项目根目录?如果是这样,从 IDE 运行它会很不方便——我需要以某种方式在其中设置路径。

更新:我尝试添加文件root.pth to package1目录内容为..。但它没有用——我猜它是有其他用途的。

结论:

  1. 始终使用绝对导入:import package1.module1

  2. 将引导程序添加到根文件夹以将某些模块作为独立脚本启动。这解决了从 IDE 运行脚本的问题,是一种 Pythonic 方法。

2007 年 4 月 22 日,布雷特·坎农 (Brett Cannon) 写道:

本 PEP 旨在改变if __name__ == "__main__": ...成语if __name__ == sys.main: ...这样你至少有机会 执行使用相对导入的包中的模块。

将此 PEP 运行到 python-ideas 上。也在那里停止了讨论 人们提出了许多新的想法。 =) 我已将它们全部列出在 被拒绝的想法部分,尽管如果有压倒性的支持 出现后,PEP 可以转向其中之一。

我对此以及任何其他建议的调整都是-1__main__机械。唯一的用例似乎是运行发生的脚本 位于模块的目录中,我一直将其视为 反模式。为了让我改变主意你必须让我相信 事实并非如此。

--吉多·范罗苏姆 http://mail.python.org/pipermail/python-3000/2007-April/006793.html


您的程序的入口点是什么?通常程序的入口点位于项目的根目录。由于它位于根目录,因此根目录中的所有模块都将是可导入的,只要有一个__init__.py文件在其中。

所以,用你的例子:

my_project/
    main.py
    package1/
        __init__.py
        module1
        module2
    package2/
        __init__.py
        module1
        module2

main.py将是您的程序的入口点。因为作为 main 执行的文件会自动放到 PYTHONPATH 上,所以package1 and package2可从顶级导入获得。

# in main.py
from package1.module1 import *
from package1.module2 import *

# in package1.module1
import module2
from package2.module1 import *

# in package2.module1 import *
import module2
from package1.module1 import *

请注意,在上面,package1 和 package2 相互依赖。情况绝对不应该如此。但这只是能够从任何地方导入的示例。

main.py也不必是什么花哨的东西。它可以非常简单:

# main.py

if __name__ == '__main__':
    from package1.module1 import SomeClass
    SomeClass().start()

我想要表达的观点是,如果一个模块需要被其他模块访问,那么该模块应该可以作为顶级导入使用。模块不应尝试将自身作为顶级导入(直接在 PYTHONPATH 上)。

这应该是相关部门的责任project如果模块直接包含在项目中,则确保可以满足所有导入。有两种方法可以做到这一点。第一种是创建引导程序文件,例如main.py在项目文件夹中。另一种方法是创建一个文件,将所有相关路径添加到 PYTHONPATH,该文件由可能存在的任何入口点加载。

例如:

# setup.py
import sys

def load():
    paths = ['/path1/','/path2/','/path3/']
    for p in path:
        sys.path.insert(0, p)

# entrypoint.py
from setup import load
load()
# continue with program

最重要的是,模块是not应该把自己放在路上。该路径应该由程序的入口点自动确定,或者由知道所有相关模块所​​在位置的安装脚本明确定义。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

导入路径 - 正确的方法? 的相关文章

随机推荐

  • 我希望 shell 脚本可执行但不可读

    我创建了一个脚本 我希望其他用户使用我们的共享系统 to 执行但不读取 我将权限设置为所有可执行文件 但撤销了读 写权限 x x x 1 dilletante staff 0 2013 04 02 11 42 expect sh 然而脚本无
  • 使用 lambda 表达式参数调用泛型方法的反射

    我正在寻找一种使用 lambda 表达式调用通用方法的方法 该表达式在项目数组中调用 Contains 在本例中 我使用实体框架Where方法 但该场景可以应用于其他IEnumerables 我需要通过 Reflection 调用上面代码的
  • 如何检查 SQL Server 当前池大小

    有没有办法检查 SQL Server 中当前连接池的大小 我不是在谈论最大连接池大小 而是当前池大小 假设最大池大小为 100 并且有 49 个打开的连接 它现在应该显示 51 个可用连接或 49 个已消耗连接 那么 有这样的查询吗 其中很
  • Golang写入套接字而不用担心数据不完整

    我们都知道 Write 方法不能保证从缓冲区中写入高字节 因此 使用原始 Write 方法将字节写入套接字的规范方法如下所示 how many bytes we have written written 0 for written lt l
  • 无法让 QWindow::fromWinId 正常工作

    我的 Qt 5 9 程序 在 X11 Linux 上 使用以下命令启动其他应用程序QProcess 我想控制这些应用程序生成的窗口 所以我获得了它们winId价值和用途QWindow fromWinId得到一个QWindow实例 问题是这些
  • Laravel $request->expectsJson()

    我正在为我的 Laravel 应用程序进行 Ajax 登录 我正在使用角度 http method POST url admin login headers Content Type application json data email
  • 如何读取图像上的文字?

    我需要将一些扫描文档解析为文本数据 是否可以使用某些软件解析图像上写的文本 如果是 请推荐任何此类在线实用程序或软件 也许一些 OCR 软件会有帮助 http en wikipedia org wiki Optical character
  • 忽略“证书未知”警报

    我有以下简单的 Python 脚本 import socket import ssl if name main s socket socket socket AF INET socket SOCK STREAM s bind 443 s l
  • 销毁 Bootstrap 弹出窗口时出现 Javascript 错误

    尝试随时更改引导程序弹出窗口的标题和内容 我遇到了一些麻烦 我在销毁选择器中的弹出窗口内容时遇到此问题 错误是这样的 TypeError undefined is not a function evaluating data option
  • T-SQL删除插入的记录

    我知道标题可能看起来很奇怪 但这就是我想做的 我有很多记录的表 我想获取其中一些记录并将它们插入到其他表中 像这样的东西 INSERT INTO TableNew SELECT FROM TableOld WHERE 棘手的部分是我希望我插
  • Jquery UI 工具提示不支持 html 内容

    今天 我将所有 jQuery 插件升级为 jQuery 1 9 1 我开始将 jQueryUI 工具提示与 jquery ui 1 10 2 一起使用 一切都很好 但是当我在内容中使用 HTML 标签时 在title我正在应用工具提示的元素
  • 我怎样才能使这个模式持久化?

    我正在寻找一种方法 让这种模式在出现后持久存在 正如此处所示 用户只需在 div 外部单击一下即可将其关闭
  • 如何制作一个反应本机输入,向用户提供验证状态反馈。 [有效、Printine、错误、编辑]

    我希望输入能够随着用户键入而不断更新 然后失去焦点 反馈将是输入周围的边框 1 Green when valid 2 Amber when typing and is in error state Green when valid 3 Re
  • 一面一示例 T 测试 Python

    在 Python 中 我使用 SciPy 进行单样本 t 检验 from scipy import stats one sample data 177 3 182 7 169 6 176 3 180 3 179 4 178 5 177 2
  • Checkstyles + Gradle 抛出引起:java.lang.IllegalArgumentException:给定名称 COMPACT_CTOR_DEF

    我最近将 checkstyle 插件添加到项目中以进行静态代码分析 但更新之后google style xml从最新的大师那里 我开始收到以下异常 org gradle api tasks TaskExecutionException Ex
  • grails 2.0 - 正确使用 serverURL 进行生产?

    Grails 2 0 改变了它使用 grails serverURL 进行开发和测试环境的方式 如manual http grails org doc 2 0 x guide single html upgradingFromPreviou
  • Python从视频文件中提取wav

    Related 如何使用python从视频文件中提取音频 https stackoverflow com questions 19216450 how to extract audio from a video file using pyt
  • 如何在 Obj-C 类别中“伪造”ivars (iPhone)

    Update iPhone OS 3 1 有关联的对象 然而 iPhone 模拟器却没有 如果您想在模拟器中测试关联的对象代码 您应该提交错误 请参阅我的问题here https stackoverflow com questions 19
  • 最接近的 3 点组

    是否有一种已知的 有效的算法来查找最接近的组three云中的点 这类似于最近点对问题 http en wikipedia org wiki Closest pair of points problem但我正在寻找三点而不是两点 Edit 最
  • 导入路径 - 正确的方法?

    我知道有很多类似或相同的问题 但我仍然无法理解 找到使用模块的正确方法 Python 是我最喜欢的语言 除了使用导入之外 我喜欢其中的所有内容 递归导入 当您尝试引用尚不存在的名称时 导入路径等 所以 我有这样的项目结构 my projec