網站首頁 編程語言 正文
文章是從stackoverflow翻譯過來的,原文地址:Relative imports for the billionth time
本文要在原理上解決 python當中相對包導入出現的問題。
問題描述
在win7、32位的電腦上,運行python2.7.3,經常會出現"Attempted relative import in non-package"這樣的問題。
為了解決這個問題,我(提問的人)搜索了以下網站,當然還有更多的網站
- http://www.python.org/dev/peps/pep-0328/
- http://docs.python.org/2/tutorial/modules.html#packages
- Python packages: relative imports
- python relative import example code does not work
- Ultimate answer to relative python imports
- Relative imports in Python
- Python: Disabling relative import
我根據pep-0328建立了以下的目錄結構
package/
__init__.py
subpackage1/
__init__.py
moduleX.py
moduleY.py
subpackage2/
__init__.py
moduleZ.py
moduleA.py
而且按照這個目錄當中的要求,建立了了spam(moduleY.py中)和eggs(moduleZ.py中)函數,現在想要在moduleX.py當中調用別的函數或模塊。很顯然,當我運行的時候,并沒有成功。
上面的第四個URL當中有如下信息,這個答案比較接近真相,但是其中的有些概念還是很難理解:
相對導入使用模塊的名稱屬性來決定模塊在包層次結構中的位置,如果模塊的名稱不包含任何包信息(例如:被設置成‘main’),那么相對導入則被解析為最頂層的位置,不管這個時候這個模塊實際上位于文件系統中的什么位置。
所以,如何讓我運行的python程序不再返回"Attempted relative import in non-package"問題?python為什么會報這個錯誤?這里的‘non-package’是什么意思?為什么以及如何去定義一個‘package’?同時解釋一下-m選項?
回答:
什么是腳本?什么是模塊?(script vs module)
直接運行一個文件和在別的文件中導入這個文件是有很大區別的,僅僅知道一個文件在目錄中的位置并不意味著python程序就認為它在什么位置。這是由python用何種方式加載(運行或者導入.run or import)這個文件來決定的。
python有兩種加載文件的方法:一種是作為頂層的腳本,另一種是當做模塊。如果你直接執行這個程序,那么這個文件就被當做是頂層腳本來執行了,在命令行里面輸入 python myfile.py 就是這個情況。如果你輸入python -m myfile.py或者在其他的文件當中使用import來導入這個文件的時候,它就被當做模塊來導入。在同一時間里,只有一個頂層腳本,頂層腳本可以這樣解釋:它是一個能夠讓你的程序從這里開始的python文件。
【文件(file)是一種無區別的叫法,如何運行和處理這個文件,決定了它的性質。直接從這個文件運行,那么這個文件就叫做腳本。導入這個文件,那么這個文件就是模塊(module)。另外,一個包(package)是一個包含有__init__.py的文件夾,下面會用到】
命名(naming)
當一個文件被加載進來,它就有一個名稱(這個名稱存儲在__name__屬性當中)。如果這個文件被當做一個頂層腳本來進行加載,那么它的名字就是__main__。如果它被當做一個模塊加載,那么它的名稱就是文件名稱,加上它所在的包名,以及所有的頂層的包名,這些名稱中間是用點號隔開的。
比如下面的例子
package/
__init__.py
subpackage1/
__init__.py
moduleX.py
moduleA.py
比如你導入moduleX(from package.subpackag1 import moduleX),它的名稱就package.subpackage1.mouleX。如果你導入moduleA的時候(from package import moduleA),它的名稱就是package.moudleA。
(注:這里是使用包導入,即把package以及里面的所有文件看做一個包,導入的時候使用from ... import ...的形式來進行,我們調用第三方包的時候就是這種情況),
但是,當你直接從命令行里面運行moduleX的時候,他的名稱則被替換為__main__。如果你直接從命令行運行moduleA,它的名稱也是__main__。當一個模塊被當做一個頂層腳本來執行的時候,它原來的名稱則會被__main__取代。
不通過包導入訪問一個模塊
這里有一個額外的問題:模塊的名稱取決于從它所在的目錄中直接導入的,還是通過包導入的。這種情況只有在該包中運行python文件,并且試圖導入這個包當中的其它文件的時候才有意義。(可以發現,上面的介紹的包導入總是在外面來訪問一個模塊)
舉個例子,如果你在package/subpackage1目錄當中打開python解釋器,然后輸入import moduleX,那么moduleX的名稱就是moduleX,而不是package.subpackage1.moduleX。這是因為python把當前的目錄添加到了搜索路徑上面。如果它發現被包含的模塊在當前的目錄當中,它將不知道該目錄也是模塊的一部分,包的信息不會出現在模塊名稱當中。
當你直接運行python解釋器(比如cmd里面輸入python,然后進入python解釋器,或者使用ipython)。在這種情況下,這種交互式的終端的名稱是__main__。
現在,你的問題有了一個關鍵性的答案:如果一個模塊名稱當中沒有點號,那么它就不會被當做是一個包。不管這個文件在磁盤的什么位置。所有關鍵在于,它的名稱是什么,而這個名稱取決于你如何加載它。
那么現在看一下你在其他的URL當中的引用的這句話:
相對導入使用模塊的名稱屬性來決定模塊在包層次結構中的位置,如果模塊的名稱不包含任何包信息(例如:被設置成‘main’),那么相對導入則被解析為最頂層的位置,不管這個時候這個模塊實際上位于文件系統中的什么位置。
相對導入...
相對導入使用模塊的名稱去決定它在一個包中的位置。當你使用了一個像這樣的相對導入:from .. import foo,這里的點號表明在包的層次結構當中上升幾個層級。比如,現在模塊的名稱是package.subpackage1.moudleX,然后..moduleA中的兩個點號表示的是上升兩個層級,到達package,然后package和moduleA結合,最終成為package.moduleA。要讓from .. import這樣的相對導入正常工作,模塊的名稱中至少要有和語句中相對應的“點”的數量。
...只能用在相對導入當使用
如果你的模塊的名稱是__main__,那么它就不被認為是在一個包當中,因為它的名稱當中不含有“點”,所以你不能在它的里面使用from .. import。如果你使用了這個語句,那么程序就會報“relative-import in non-package"錯誤。
腳本不能包含相對導入:
當你直接運行moduleX或者是在命令行終端里運行程序的時候,這個時候模塊的名稱都是__main__,這就表明你不能使用相對導入。因為他們的名稱表示他們并不在一個包當中。注:當你運行的python目錄就是你模塊所在的目錄的時候,上面這種情況也會發生,這種情況下python過早的尋找當前目錄的模塊,并沒有認為他們也是包的一部分。
當你運行交互式的解釋器的時候,交互式進程的名稱永遠是__main__,因此你不能在交互式進程當中使用相對導入。相對導入只能在模塊文件當中使用。
兩個解決方法:
1:如果你想直接運行moduleX,但是你又想把它當做一個包的一部分,你可以使用python -m package.subpackage.moduleX. -m參數告訴python把它當做一個模塊來加載,而不是頂層的腳本。
2:或許你并不想直接運行moduleX,你想在其它的腳本當中使用moduleX的函數,比如說這個腳本是myfile.py。如果是這種情況,需要把myfile.py放在別的地方,而不是在package目錄里面。在myfile.py使用一下語句就可以正常工作了:from package.moduleA import spam.
注:對于上面說的這兩種情況,包目錄(比如上面的package)必須存在于python的搜索路徑下面(sys.path)。如果不存在,你將不能夠使用包中的任何東西。
自從python2.6,模塊的名稱不在決定使用__name__屬性,而是使用__packege__屬性。這就是為什么我避免使用__name__這么明確的名稱來代表一個模塊的名稱。自從python2.6,一個模塊的名稱是由__package__+'.'+__name__來確定的,如果__packege__是None的話,那么這個名稱就是__name__了。
原文鏈接:https://www.cnblogs.com/jiaxin359/p/7580375.html
- 上一篇:沒有了
- 下一篇:沒有了
相關推薦
- 2022-10-01 SQL中concat和substr組合運用解析_MsSql
- 2022-02-24 JDBC中在結果集中以列順序獲取值時注意類型匹配
- 2022-07-28 C++超詳細講解逗號操作符_C 語言
- 2022-06-22 Python?Tkinter?GUI編程實現Frame切換_python
- 2024-03-14 Linux Nginx自定義安裝目錄
- 2024-07-18 Spring Security之安全異常處理
- 2022-07-22 px和em和rem的區別
- 2023-07-09 【elementplus】body設置zoom后,el-table開啟show-overflow-t
- 欄目分類
-
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支