我想从同一目录中的另一个文件导入一个函数。

通常,以下工作之一:

from .mymodule import myfunction
from mymodule import myfunction

…但另一个给了我一个错误:

ImportError: attempted relative import with no known parent package
ModuleNotFoundError: No module named 'mymodule'
SystemError: Parent module '' not loaded, cannot perform relative import

这是为什么?


当前回答

我有一个类似的问题,并通过在工作目录中创建一个指向包的符号链接来解决:

在-s../../..中/我的包我的包

然后照常导入:

导入my_package

我知道这更像是“Linux”解决方案,而不是“Python”解决方案。但这仍然是一种有效的方法。

其他回答

我需要从主项目目录运行python3才能使其正常工作。

例如,如果项目具有以下结构:

project_demo/
├── main.py
├── some_package/
│   ├── __init__.py
│   └── project_configs.py
└── test/
    └── test_project_configs.py

解决方案

我将在文件夹project_demo/中运行python3,然后执行

from some_package import project_configs

我正在获取此ImportError:尝试在没有已知父包的情况下进行相对导入

在我的程序中,我使用当前路径中的文件导入其函数。

from .filename import function

然后我用包名修改了当前路径(Dot)。这解决了我的问题。

from package_name.filename import function

我希望上面的答案对你有所帮助。

我的样板,以使包中的模块具有可独立运行的相对导入。

包/模块.py

## Standalone boilerplate before relative imports
if __package__ is None:                  
    DIR = Path(__file__).resolve().parent
    sys.path.insert(0, str(DIR.parent))
    __package__ = DIR.name

from . import variable_in__init__py
from . import other_module_in_package
...

现在,您可以以任何方式使用模块:

照常运行模块:python-m package.module将其用作模块:python-c“来自包导入模块”独立运行:python package/module.py或者使用shebang(#!/bin/env-python):package/module.py

NB!如果模块与包同名,则使用sys.path.append而不是sys.path.insert将导致难以跟踪的错误。例如my_script/my_script.py

当然,如果您的包层次结构中有较高级别的相对导入,那么这是不够的,但在大多数情况下,这是可以的。

从同一目录导入

首先,您可以从同一目录导入。

这是文件结构。。。

Folder
 |
 ├─ Scripts
 |   ├─ module123.py
 |
 ├─ main.py
 ├─ script123.py

这是main.py

from . import script123
from Scripts import module123

如您所见,从导入。从当前目录导入。

注意:如果使用IDLE以外的任何方法运行,请确保在运行之前将终端导航到与main.py文件相同的目录。

此外,从本地文件夹导入也有效。

从父目录导入

正如我在GitHub中看到的,有以下方法。

采用以下文件树。。。

ParentDirectory
 ├─ Folder
 |   |
 |   ├─ Scripts
 |   |   ├─ module123.py
 |   |
 |   ├─ main.py
 |   ├─ script123.py
 |
 ├─ parentModule.py

然后,只需将此代码添加到main.py文件的顶部。

import inspect
import os
import sys

current_dir = os.path.dirname(os.path.abspath(inspect.getfile(inspect.currentframe())))
parent_dir = os.path.dirname(current_dir)
sys.path.insert(0, parent_dir)

from ParentDirectory import Stuff

我在尝试编写一个可以作为模块或可执行脚本加载的python文件时遇到了类似的问题。

安装程序

/path/to/project/
├── __init__.py
└── main.py
    └── mylib/
        ├── list_util.py
        └── args_util.py

具有:

main.py:

#!/usr/bin/env python3
import sys
import mylib.args_util

if __name__ == '__main__':
    print(f'{mylib.args_util.parseargs(sys.argv[1:])=}')

mylib/list_util.py:

def to_int_list(args):
    return [int(x) for x in args]

mylib/args_util.py:

#!/usr/bin/env python3
import sys
from . import list_util as lu

def parseargs(args):
    return sum(lu.to_int_list(args))

if __name__ == '__main__':
    print(f'{parseargs(sys.argv[1:])=}')

输出

$ ./main.py 1 2 3
mylib.args_util.parseargs(sys.argv[1:])=6

$ mylib/args_util.py 1 2 3
Traceback (most recent call last):
  File "/path/to/project/mylib/args_util.py", line 10, in <module>
    from . import list_util as lu
ImportError: attempted relative import with no known parent package

解决方案

我决定使用Bash/Python多语言解决方案。Bash版本的程序只调用python3-m mylib.args_util,然后退出。

Python版本忽略Bash代码,因为它包含在docstring中。

Bash版本忽略Python代码,因为它使用exec停止解析/运行行。

mylib/args_util.py:

#!/bin/bash
# -*- Mode: python -*-
''''true
exec /usr/bin/env python3 -m mylib.args_util "$@"
'''

import sys
from . import list_util as lu

def parseargs(args):
    return sum(lu.to_int_list(args))

if __name__ == '__main__':
    print(f'{parseargs(sys.argv[1:])=}')

输出

$ ./main.py 1 2 3
mylib.args_util.parseargs(sys.argv[1:])=6

$ mylib/args_util.py 1 2 3
parseargs(sys.argv[1:])=6

解释

第1行:#/bin/bash;这是“shebang”线;它告诉交互式shell如何运行该脚本。Python:忽略(注释)Bash:忽略(注释)第2行:#-*-模式:python-*-可选;这被称为“模式线”;它告诉Emacs使用Python语法高亮显示,而不是在读取文件时猜测语言是Bash。Python:忽略(注释)Bash:忽略(注释)第3行:“”“truePython:将其视为以“true”开头的未分配文档字符串\nBash:将其视为三个扩展为true的字符串(其中前两个是空字符串)(即“+”+“true”=“true”);然后它运行为true(它什么都不做)并继续到下一行第4行:exec/usr/bin/env python3-m mylib.args_util“$@”Python:仍然将其视为第3行中文档字符串的一部分。Bash:运行python3-m mylib.args_util,然后退出(它不会分析超出这一行的任何内容)第5行:“”Python:将其视为第3行中文档字符串的结尾。Bash:不解析此行

注意事项

这在Windows上不起作用:解决方法:使用WSL或批处理包装脚本调用python-m mylib.args_util。仅当当前工作目录设置为/path/to/project/时,此操作才有效。解决方法:调用/usr/bin/env时设置PYTHONPATH#!/bin/bash#-*-模式:python-*-“”“真的执行/usr/bin/env python3\PYTHONPATH=“$(cd”$(dirname“$0”)/。。“;pwd)”\-m mylib.args_util“$@”'''