在python项目中使用相对路径读取文件

Posted

技术标签:

【中文标题】在python项目中使用相对路径读取文件【英文标题】:Reading file using relative path in python project 【发布时间】:2022-01-17 21:11:46 【问题描述】:

假设我有一个结构如下的 python 项目:

project
    /data
        test.csv
    /package
        __init__.py
        module.py
    main.py

__init__.py:

from .module import test

module.py:

import csv

with open("..data/test.csv") as f:
    test = [line for line in csv.reader(f)]

main.py:

import package

print(package.test)

当我运行main.py 时出现以下错误:

 C:\Users\Patrick\Desktop\project>python main.py
Traceback (most recent call last):
  File "main.py", line 1, in <module>
    import package
  File "C:\Users\Patrick\Desktop\project\package\__init__.py", line 1, in <module>
    from .module import test
  File "C:\Users\Patrick\Desktop\project\package\module.py", line 3, in <module>
    with open("../data/test.csv") as f:
FileNotFoundError: [Errno 2] No such file or directory: '../data/test.csv'

但是,如果我从 package 目录运行 module.py,则不会出现任何错误。所以似乎open(...) 中使用的相对路径仅相对于原始文件的运行位置(即__name__ == "__main__")?我不想使用绝对路径。有什么方法可以解决这个问题?

【问题讨论】:

作为旁注,引用 PEP8:“非常不鼓励使用相对导入进行包内导入。始终对所有导入使用绝对包路径。” 这里,from package.module import test 【参考方案1】:

相对路径相对于current working directory。 如果你不希望你的路径是,它必须是绝对的。

但是有一个常用的技巧可以从当前脚本构建绝对路径:使用其__file__ 特殊属性:

from pathlib import Path

path = Path(__file__).parent / "../data/test.csv"
with path.open() as f:
    test = list(csv.reader(f))

这需要 python 3.4+(对于pathlib 模块)。

如果您仍需要支持旧版本,您可以通过以下方式获得相同的结果:

import csv
import os.path

my_path = os.path.abspath(os.path.dirname(__file__))
path = os.path.join(my_path, "../data/test.csv")
with open(path) as f:
    test = list(csv.reader(f))

[2020 年编辑:python3.4+ 现在应该是常态,所以我首先移动了受 jpyams 评论启发的 pathlib 版本]

【讨论】:

当然,如果你使用 Python 3.4+,你可以直接使用pathlib.Path:Path(__file__).parent.resolve()【参考方案2】:

对于 Python 3.4+:

import csv
from pathlib import Path

base_path = Path(__file__).parent
file_path = (base_path / "../data/test.csv").resolve()

with open(file_path) as f:
    test = [line for line in csv.reader(f)]

【讨论】:

【参考方案3】:

这对我有用。

with open('data/test.csv') as f:

 

【讨论】:

【参考方案4】:

我的 Python 版本是 Python 3.5.2,接受的答案中提出的解决方案对我不起作用。我仍然收到错误提示

FileNotFoundError: [Errno 2] No such file or directory

当我从终端运行my_script.py 时。虽然当我通过 PyCharm IDE(PyCharm 2018.3.2(社区版))的运行/调试配置运行它时它运行良好。

解决方案

而不是使用:

my_path = os.path.abspath(os.path.dirname(__file__)) + some_rel_dir_path 

按照接受的答案中的建议,我使用了:

my_path = os.path.abspath(os.path.dirname(os.path.abspath(__file__))) + some_rel_dir_path

说明: 将 os.path.dirname(__file__) 更改为 os.path.dirname(os.path.abspath(__file__)) 解决了以下问题:

当我们这样运行脚本时:python3 my_script.py __file__ 变量只有一个字符串值“my_script.py”,没有通向该特定脚本的路径。这就是为什么方法dirname(__file__) 返回一个空字符串“”。这也是为什么my_path = os.path.abspath(os.path.dirname(__file__)) + some_rel_dir_path 实际上与my_path = some_rel_dir_path 相同的原因。因此在尝试使用open 方法时会给出FileNotFoundError: [Errno 2] No such file or directory,因为没有像“some_rel_dir_path”这样的目录。

从 PyCharm IDE 运行/调试配置运行脚本有效,因为它运行命令 python3 /full/path/to/my_script.py(其中“/full/path/to”由我们在运行/调试配置的“工作目录”变量中指定),而不仅仅是python3 my_script.py 就像我们从终端运行它一样。

希望这会有用。

【讨论】:

这确实是一个警告。显式运行脚本是如此罕见,以至于您似乎是第一个注意到的(通常您会将#!/usr/bin/env python 放在顶部,将其标记为可执行并以./myscript.py 运行它)。 pathlib.Path 版本没有这个问题,但它可能是自 python3.4 以来更好的选择。【参考方案5】:

试试

with open(f"os.path.dirname(sys.argv[0])/data/test.csv", newline='') as f:

【讨论】:

【参考方案6】:

当下面的代码运行时,我惊呆了。

import os

for file in os.listdir("../FutureBookList"):
    if file.endswith(".adoc"):
        filename, file_extension = os.path.splitext(file)
        print(filename)
        print(file_extension)
        continue
    else:
        continue

所以,我检查了documentation,它说:

3.6 版更改:接受类似路径的对象。

path-like object:

表示文件系统路径的对象。类似路径的对象是 str 或...

我做了一个little more digging,以下也有效:

with open("../FutureBookList/file.txt") as file:
   data = file.read()

【讨论】:

以上是关于在python项目中使用相对路径读取文件的主要内容,如果未能解决你的问题,请参考以下文章

如何读取java项目中文件的相对路径

Vscode的相对路径读取问题及处理

IDEA 相对路径读取项目中的文件

Java FileReader使用相对路径读取文件

kotlin关于读取文件路径问题

如何读取文本文件的相对路径