问题描述

近日在尝试引用其他文件的代码时,遇到了错误: ImportError: attempted relative import with no known parent package.

问题大致是这样的:我想在 code2.py 中引用 code1.py 的函数,如 from ..folder1.code1 import xxx,运行 code2.py 时出现错误。

root
├── folder1
│ └── code1.py
├── folder2
│ └── code2.py
└── main.py

太长不看版

如果你要在 code2.py 中引用 code1.py 的函数,那么可以:

  • 改变文件结构,考虑在 main.py 中调用,运行 main.py
  • code2.py 中增加 root 的位置到搜索路径 sys.path.append, 代码使用 from folder1.code1 import xxx
  • -m 选项运行: python -m root.folder2.code2,代码可以使用 from folder1.code1 import xxxfrom ..folder1.code1 import xxx [我认为这是最优解!]

详细解释

如果对导入的概念不是很理解的话,可能会遇到:

  • ModuleNotFoundError: No module named 'xxx'
  • ImportError: attempted relative import with no known parent package

首先明确两种导入方法:

  1. from xxx import yyy 则是从已知的模块导入
  2. “relative import” 即 from .xxx import yyy,根据从当前文件的相对路径导入。

第一种方法

具体可参考官方文档 the-module-search-path

仅适用于模块(文件夹)或脚本(文件)存在于搜索路径中,导入时,Python 解释器会首先搜索内置模块,如果没有,则去以下三个位置搜索:

  1. 当前文件所在目录
  2. 环境变量 PYTHONPATH 指定的目录
  3. Python 默认的安装目录

可以查看 sys.path,显然,当前运行脚本所在的文件夹被放在了搜索路径的首位,因此该文件夹下的所有内容均可被引入。

import sys
print(sys.path)
# ['/.../path-to-this-folder', '/usr/lib/python310.zip', '/usr/lib/python3.10', '/usr/lib/python3.10/lib-dynload', '/home/thor/.local/lib/python3.10/site-packages', '/usr/local/lib/python3.10/dist-packages', '/usr/lib/python3/dist-packages']

要解决开头提出的问题,即引入其他文件夹下的内容,可以把 root 的位置添加到搜索路径中:(好吧,这样很不优雅……)

import sys
sys.path.join("/path/to/root") # 用绝对路径,需要从根目录开始
sys.path.join("..") # 用相对路径,但是命令行当前位置不能出错 from folder1.code1 import xxx

可以参考这段代码[1]

if __package__:
from .. import config
else:
sys.path.append(os.dirname(__file__) + '/..')
import config

第二种方法

具体可参考官方文档 packages

需要明确的是,这种方法只适用于 package 内部!

当你把 code2.py 作为脚本运行时,即 python code2.py,此时 python 并不会认为它属于某一个 package, 即使存在 __init__.py。可以 print(__package__) 进行验证,作为脚本运行时为 None,否则则应该为 xxx.yyy 的形式[1:1]

(网络上有很多地方都说添加 __init__.py 就可以解决问题,但事实是并不会 ,在我的测试中,在本文提到的所有的解决方法中,添加 __init__.py 与否似乎不会带来什么影响。)

因此,开头描述的问题中,要使用相对导入的形式在 code2.py 中引用 code1.py 的代码,必须使用:

python -m root.folder1.code1
  • 这里把 root 及其内部当作一个完整的 package,而 package 内的脚本可以使用相对导入互相引用。
  • 这里不带 .py 后缀。
  • 不可以为 python -m folder1.code1,此时把 folder1 及其内部当作一个完整的 package, 无法引用到以外的内容,会遇到 ImportError: attempted relative import beyond top-level package

除了命令行调用时进行调整,在脚本中 import 也是一样的道理:

newroot
├── root
│   ├── folder1
│   │   └── code1.py
│   ├── folder2
│   │   └── code2.py
│   └── main.py
└── upper_main.py
  • upper_main.py 中添加 from root.folder2 import code2 并运行时,它会把 root 当作一个包,此时code2.py中的 from ..folder1.code1 import xxx 可以正常执行
  • main.py 中添加 import folder2.code2 并运行时,它会把 folder2 当作一个包,此时 code2.py 中的 from .xx import 可以正常执行,而 from ..folder1.code1 import xxx 会遇到 ImportError: attempted relative import beyond top-level package.

其他

说明:

  • 这里仅说明我尝试成功得出的经验,不排除有其他正确做法。
  • 我还看到过类似 code2.py 中有 from folder1 import code1 这种做法,没有测试过其适用条件,不过模块内部感觉使用相对引用比较好。

其他参考资料:


  1. 3 Best Solutions For "ImportError: attempted relative import with no known parent package" [2021 Updated]

Python 引用问题 - ImportError: attempted relative import with no known parent package的更多相关文章

  1. Python工程:ImportError: attempted relative import with no known parent package

    Python工程:ImportError: attempted relative import with no known parent package 解决方法: 1.对每个目录创建的时候都选择创建 ...

  2. 执行代码出现ImportError:attempted relative import with no known parent package

    前言 在这篇文章中,我将会解析 ImportError: attempted relative import with no known parent package 这个异常的原因.当你在运行的py ...

  3. 执行文件异常报错:ImportError: attempted relative import with no known parent package

    这个问题困扰了我很久了,网上的解决方法都很一致,找来找去都是一样的解决方法,在导入包的文件和执行文件加入 1 print('__file__={0:<35} | __name__={1:< ...

  4. ImportError: attempted relative import with no known parent package

    或者检查所导包是否存在__init__.py文件,没有则添加上即可使当前文件夹变为包.

  5. python相对包导入报“Attempted relative import in non-package”错误

    文章是从stackoverflow翻译过来的,原文地址:Relative imports for the billionth time 本文要在原理上解决  python当中相对包导入出现的问题. 问 ...

  6. Python ValueError: Attempted relative import in non-package Relative import相对引用 错误

    包含相对路径import的python脚本不能直接运行,只能作为module被引用. 例如 from . import mod1 有这样代码的文件只能最为moulule为不能直接运行.相对路径就是相对 ...

  7. python:Attempted relative import in non-package

    problem:Attempted relative import in non-package 所谓相对路径其实就是相对于当前module的路径,但如果直接执行脚本,这个module的name就是“ ...

  8. Python 相对导入attempted relative import beyond top-level package

    ValueError: attempted relative import beyond top-level package 假设有如下层次包目录 project/ __init__.py mypac ...

  9. ValueError: Attempted relative import in non-package

    执行:python deom/scripts/populate.py ValueError: Attempted relative import in non-package solve:python ...

  10. Python导入自定义类时显示错误:attempted relative import beyond top-level package

    显示这个错误可能有两个原因: 1.文件夹中没有包含__init__.py文件,该文件可以为空,但必须存在该文件. 2.把该文件当成主函数入口,该文件所在文件夹不能被解释器视作package,所以可能导 ...

随机推荐

  1. day26:装饰器&面向对象当中的方法&property

    目录 1.装饰器 1.1 装饰器的基本用法 1.2 @符号的使用 1.3 装饰器的嵌套 1.4 用装饰器扩展带有参数的原函数 1.5 用装饰器扩展带有参数和返回值的原函数 1.6 用类装饰器扩展原函数 ...

  2. Yii2批量插入数据

    方法一 yii2一次插入多行数据  /** * @inheritdoc 批量添加 * @params $add array 添加数据 */ public function add_all($add) ...

  3. for of 和 for in 的区别

    1 var arr = ["f", "6", 3, "a", 7]; 2 var obj = { name: "shun" ...

  4. vue中使用svg并设置大小

    1.安装依赖 npm install --save-dev svg-sprite-loader 2. 新建svg资源文件夹     src/assets/svg 将svg资源放入此目录,接下来会在配置 ...

  5. 【Docker】容器管理

    一.容器生命周期及启动过程 1.容器生命周期 2.容器启动过程 二.容器管理命令 Usage: docker [OPTIONS] COMMAND A self-sufficient runtime f ...

  6. 2022-06-01:给定一个数组arr,可能有正、有负、有0,无序。 只能挑选两个数字,想尽量让两个数字加起来的绝对值尽量小。 返回可能的最小的值。

    2022-06-01:给定一个数组arr,可能有正.有负.有0,无序. 只能挑选两个数字,想尽量让两个数字加起来的绝对值尽量小. 返回可能的最小的值. 答案2022-06-01: 排序,双指针. 代码 ...

  7. 2021-05-27:定义何为step sum?比如680,680+68+6=754,680的step sum叫754。

    2021-05-27:定义何为step sum?比如680,680+68+6=754,680的step sum叫754.给定一个整数num,判断它是不是某个数的step sum? 福大大 答案2021 ...

  8. Winform 遮罩懒人处理法

    前言 之前有个项目需要执行一个略微耗时的操作大概五六七八九十秒这样子,这个时候程序不能做其他操作,只能等待操作完成.为了提升一丝使用体验同时让Winform程序看上去高级一点,就想到加一个遮罩层(Ma ...

  9. ChatGPT 推出 iOS 应用,支持语音输入,使用体验如何?

    最近,OpenAI 宣布推出官方 iOS 应用,允许用户随时随地访问其高人气 AI 聊天机器人,此举也打破了近几个月内苹果 App Store 上充斥似是而非的山寨服务的窘境. 该应用程序是 Chat ...

  10. MAX30102采集心率数据

    一个100行的代码调试都可能会让程序员遇到很多挫折,所以,面对挫折,我们永远不能低头. 关于MAX30102驱动配置程序,网上搜索博客有一堆资料,c/c++写的驱动代码都有, 可参考博客: MAX30 ...