1. 绝对路径引入
Python 在搜索模块时,依次搜索sys.path
里的位置,直到找到模块为止。下面命令可以查看当前的搜索路径:
import sys print(sys.path)
sys.path
的初始值来源于两个(其实还有一些更复杂但不常用的)。一个是系统的PYTHONPATH
变量,因此可通过设置该变量,来设置 Python 默认的搜索位置。比如:
export PYTHONPATH=/opt/python:$PYTHONPATH echo $PYTHONPATH
将该命令放在系统初始化脚本(/etc/environment)
或者 BASH 初始化脚本(/~/.bashrc)
里,可以对每个新开的窗口有效。
sys.path
的另一个来源是当前执行程序所在的目录 (而不是当前目录)。比如当前目录下文件夹./cc
下有一个b.py
,那么执行./cc/b.py
时,./cc
(而不是./!
)将被加到sys.path
:
python ./cc/b.py
2. 相对路径引用
上面说的是搜索模块都是指绝对路径引用。对于非系统目录,就需要操纵sys.path。但操纵sys.path有外溢效果,因为它是一个全局变量。对于同一个库里的模块的互相引用,可以考虑使用相对路径:
from . import abc from .abc import fool from ..up import foo
但相对路径有两个很恶心的问题,使得用法极为受限。其中一个是:
Note that both explicit and implicit relative imports are based on the name of the current module. Since the name of the main module is always __main__, modules intended for use as the main module of a Python application should always use absolute imports.
包含相对路径 import 的 python 脚本不能直接运行,只能作为 module 被引用。原因正如手册中描述的,所谓相对路径其实就是相对于当前 module 的路径,但如果直接执行脚本,这个 module 的 name 就是__main__, 而不是 module 原来的 name , 这样相对路径也就不是原来的相对路径了,导入就会失败。
在使用相对引用的文件中,不能有 __main__ 方法,只执行作为一个 module 进行引用,而不是直接执行脚本。
举个简单例子。假设./cc/
目录下已有一个./cc/b.py
(内容为空)。当前目录下的./a.py
内容为:
from .cc import b
那么直接运行python ./a.py
将会报错:
ModuleNotFoundError: No module named '__main__.cc'; '__main__' is not a package
另一个是常见的错误是: ValueError: attempted relative import beyond top-level package。
在涉及到相对导入时,package所对应的文件夹必须正确的被python解释器视作package,而不是普通文件夹。否则由于不被视作package,无法利用package之间的嵌套关系实现python中包的相对导入。
文件夹被python解释器视作package需要满足两个条件:
1、文件夹中必须有__init__.py文件,该文件可以为空,但必须存在该文件。
2、不能作为顶层模块来执行该文件夹中的py文件(即不能作为主函数的入口)。
补充:在"from YY import XX"这样的代码中,无论是XX还是YY,只要被python解释器视作package,就会首先调用该package的__init__.py文件。如果都是package,则调用顺序是YY,XX。
另外,练习中“from . import XXX”和“from .. import XXX”中的'.'和'..',可以等同于linux里的shell中'.'和'..'的作用,表示当前工作目录的package和上一级的package。
举个例子:
testIm/ --__init__.py --main.py : from Tom import tom --Tom/ --__init__.py : print("I'm Tom's __init__!") --tom.py : from . import tomBrother, from .. import kate,print("I'm Tom!") --tomBrother.py print(I'm Tom's Brother!) --Kate/ --__init__.py : print("I'm Kate's __init__!") --kate.py
运行文件:main.py
结果:
I'm Tom's __init__!
I'm Tom's Brother!
Traceback (most recent call last):
File "D:\PythonLearning\TestIm\main.py", line 3, in <module>
from Tom import tom
File "D:\PythonLearning\TestIm\Kate\kate.py", line 4, in <module>
from .. import kate
ValueError: attempted relative import beyond top-level package
>
可以看到from . import tomBrother顺利执行,首先执行了Tom文件夹下的__init__.py文件,后来执行了tomBrother.py文件,但是当执行到“from .. import kate”时报错,这是因为我们是在TestIm文件夹下把main.py文件作为主函数的入口执行的,因此尽管TestIm文件夹中有__init__.py文件,但是该文件夹不能被python解释器视作package,即Tom package不存在上层packge,自然会报错,相对导入时超出了最高层级的package。
修改方法:
test/ --main.py : from testIm.Tom import tom --testIm/ --__init__.py --Tom/ --__init__.py : print("I'm Tom's __init__!") --tom.py : from . import tomBrother, from .. import Kate,print("I'm Tom!") --tomBrother.py print(I'm Tom's Brother!) --Kate/ --__init__.py : print("I'm Kate's __init__!") --kate.py
运行文件:main.py
结果:
I'm top's __init__!
I'm Tom's __init__!
I'm Tom's Brother!!
I'm Kate's __init__!
I'm Tom!
即主函数入口不在TestIm中,则TestIm和其同样包含__init__.py文件的子文件夹都被python解释器视作package,形成相应的嵌套关系。可以正常使用from . import XXX和from .. import XXX。
以上就是详解Python中的路径问题的详细内容,更多关于Python 路径的资料请关注其它相关文章!
免责声明:本站文章均来自网站采集或用户投稿,网站不提供任何软件下载或自行开发的软件! 如有用户或公司发现本站内容信息存在侵权行为,请邮件告知! 858582#qq.com
RTX 5090要首发 性能要翻倍!三星展示GDDR7显存
三星在GTC上展示了专为下一代游戏GPU设计的GDDR7内存。
首次推出的GDDR7内存模块密度为16GB,每个模块容量为2GB。其速度预设为32 Gbps(PAM3),但也可以降至28 Gbps,以提高产量和初始阶段的整体性能和成本效益。
据三星表示,GDDR7内存的能效将提高20%,同时工作电压仅为1.1V,低于标准的1.2V。通过采用更新的封装材料和优化的电路设计,使得在高速运行时的发热量降低,GDDR7的热阻比GDDR6降低了70%。
更新日志
- 《暗喻幻想》顺风耳作用介绍
- 崔健1985-梦中的倾诉[再版][WAV+CUE]
- 黄子馨《追星Xin的恋人们2》HQ头版限量编号[WAV+CUE]
- 孟庭苇《情人的眼泪》开盘母带[低速原抓WAV+CUE]
- 孙露《谁为我停留HQCD》[低速原抓WAV+CUE][1.1G]
- 孙悦《时光音乐会》纯银CD[低速原抓WAV+CUE][1.1G]
- 任然《渐晚》[FLAC/分轨][72.32MB]
- 英雄联盟新英雄安蓓萨上线了吗 新英雄安蓓萨技能介绍
- 魔兽世界奥杜尔竞速赛什么时候开启 奥杜尔竞速赛开启时间介绍
- 无畏契约CGRS准星代码多少 CGRS准星代码分享一览
- 张靓颖.2012-倾听【少城时代】【WAV+CUE】
- 游鸿明.1999-五月的雪【大宇国际】【WAV+CUE】
- 曹方.2005-遇见我【钛友文化】【WAV+CUE】
- Unity6引擎上线:稳定性提升、CPU性能最高提升4倍
- 人皇Sky今日举行婚礼!电竞传奇步入新篇章