網(wǎng)站首頁 編程語言 正文
楔子
導(dǎo)入一個模塊,我們一般都會使用 import 關(guān)鍵字,但有些場景下 import 難以滿足我們的需要。所以除了 import 之外還有很多其它導(dǎo)入模塊的方式,下面就來介紹一下。
__import__
這是一個內(nèi)置函數(shù),解釋器在 import 的時候,實際上就執(zhí)行了這個函數(shù)。
#?import?os?等價于如下方式
os?=?__import__("os")
print(os)??#?<module?'os'?from?'C:\\python38\\lib\\os.py'>
#?但是這種方式不能多級導(dǎo)入
path?=?__import__("os.path")
print(path)??#?<module?'os'?from?'C:\\python38\\lib\\os.py'>
#?可以看到,導(dǎo)入的仍是?os,而不是 os.path
#?如果想導(dǎo)入子模塊,需要一個參數(shù)?fromlist
#?我們給它傳一個非空列表即可
path?=?__import__("os.path",?fromlist=[""])
print(path)??#?<module?'ntpath'?from?'C:\\python38\\lib\\ntpath.py'>
但是官方不建議使用這個函數(shù),因為它是專門給解釋器用的,我們可以使用一個模塊。
import?importlib
os?=?importlib.import_module("os")
print(os)??#?<module?'os'?from?'C:\\python38\\lib\\os.py'>
#?可以多級導(dǎo)入
path?=?importlib.import_module("os.path")
print(path)??#?<module?'ntpath'?from?'C:\\python38\\lib\\ntpath.py'>
所以當(dāng)導(dǎo)入的模塊名以字符串的形式存在時,或者模塊名不符合規(guī)范時,就可以使用這種方式。
importlib.machinery
importlib.machinery 里面提供了三種 Loader,可以讓我們以打開文件的方式導(dǎo)入一個模塊。
from?importlib.machinery?import?(
????SourceFileLoader,??#?導(dǎo)入源文件
????SourcelessFileLoader,??#?導(dǎo)入 pyc?文件
????ExtensionFileLoader??#?導(dǎo)入擴(kuò)展文件
)
#?參數(shù)一:給模塊起個名字
#?參數(shù)二:文件路徑
os?=?SourceFileLoader(
????"我是?os?模塊",
????r"C:\python38\lib\os.py"
).load_module()
print(os)
"""
<module?'我是?os?模塊'?from?'C:\\python38\\lib\\os.py'>
"""
print(os.path.join("video",?"overwatch",?"hanzo.mp4"))
"""
video\overwatch\hanzo.mp4
"""
#?我們看到結(jié)果一切正常,但有一點需要注意
#?如果是導(dǎo)入包的話,那么要導(dǎo)入包里面的?__init__.py?文件
pd?=?SourceFileLoader(
????"我是?pandas?模塊",
????r"C:\python38\lib\site-packages\pandas\__init__.py"
).load_module()
print(pd.DataFrame({"a":?[1,?2,?3],?"b":?[4,?5,?6]}))
"""
???a??b
0??1??4
1??2??5
2??3??6
"""
#?如果只寫到?pandas,那么會拋出?PermissionError
#?因為我們不能把目錄當(dāng)成文件來讀取
#?至于?import?一個包,本質(zhì)上也是加載包內(nèi)部的?__init__.py?
#?但這里需要我們顯式地加上?__init__.py
同理加載 pyc 和 pyd 也是類似的,但需要注意的是,加載普通文件和 pyc 文件時,我們可以隨便起名字,也就是第一個參數(shù)任意。但對于 pyd 文件,第一個參數(shù)必須和 pyd 文件的名字保持一致。
通過 module 類創(chuàng)建模塊
Python 一切皆對象,模塊自然也不例外。既然是對象,那么必然就會有相應(yīng)的類來實例化它。
import?os
import?hashlib
import?numpy
#?os.__class__?等價于?type(os)
print(os.__class__)??#?<class?'module'>
print(hashlib.__class__)??#?<class?'module'>
print(numpy.__class__)??#?<class?'module'>
在 Python 里面,我們一般會把單獨的可導(dǎo)入文件稱之為模塊,把包含多個模塊的目錄稱之為包。通過模塊和包,我們可以對項目進(jìn)行功能上的劃分,分門別類地進(jìn)行組織。
但不管是模塊、還是包,它們都是 module 這個類的實例對象,打印結(jié)果也能說明這一點。所以從解釋器的角度來看的話,模塊和包區(qū)分的并沒有那么明顯,直接把包看做是包內(nèi)部的 __init__.py 即可。
既然模塊的類型是 class module,那么我們是不是也可以通過調(diào)用類型對象的方式創(chuàng)建呢?顯然是可以的,但是 module 這個類解釋器沒有暴露給我們,直接用的話會提示變量 module 未定義。所以只能先隨便導(dǎo)入一個模塊,然后通過 type 函數(shù)或者 __class__ 屬性獲取。
#?不過?types?模塊內(nèi)部已經(jīng)幫我們做好了
#?ModuleType?=?type(sys)
from?types?import?ModuleType
print(ModuleType)??#?<class?'module'>
#?類對象有了,下面就可以創(chuàng)建了
#?module?類接收兩個參數(shù)
#?參數(shù)一:模塊的名字,必須傳遞
#?參數(shù)二:模塊的 doc,不傳默認(rèn)為 None
satori?=?ModuleType("古明地覺",?"模塊的名字是一個女孩,她來自地靈殿")
print(satori)??#?<module?'古明地覺'>
print(satori.__doc__)??#?模塊的名字是一個女孩,她來自地靈殿
#?但此時模塊里面是沒啥東西的,我們加一些屬性吧
#?操作模塊本質(zhì)上是在操作它的屬性字典
code?=?"""
age?=?16
def?foo():
????return?"^_^"
"""
#?執(zhí)行?code,結(jié)果會體現(xiàn)在?satori?的屬性字典中
exec(code,?satori.__dict__)
print(satori.age)??#?16
print(satori.foo())??#?^_^
需要注意的是里面 exec 函數(shù),它會把字符串當(dāng)成代碼來執(zhí)行,所以這就要求字符串的來源必須是可靠的,我們能夠確保不會出現(xiàn)惡意內(nèi)容。而如果是用戶傳遞的字符串,那么絕不能用 exec 來執(zhí)行,當(dāng)然 eval 也是同理。
然后是 exec 的第二個參數(shù),表示執(zhí)行時的名字空間,默認(rèn)是全局名字空間。所以當(dāng)不指定第二個參數(shù)時,exec(code) 相當(dāng)于創(chuàng)建了兩個全局變量:age 和 foo。
code?=?"""
age?=?16
def?foo():
????return?"^_^"
"""
exec(code)
print(age)??#?16
print(foo())??#?^_^
但是我們在執(zhí)行的時候,將它換成了 satori.__dict__,所以結(jié)果相當(dāng)于給模塊添加了兩個變量,或者說屬性。
將一個類的實例變成一個模塊
如果想將一個類的實例變成模塊,那么這個類應(yīng)該繼承 ModuleType。
import?sys
from?types?import?ModuleType
class?A(ModuleType):
????def?__init__(self,?module_name):
????????super().__init__(module_name)
????def?__getattr__(self,?item):
????????return?f"不存在的屬性:?{item}"
????def?__setattr__(self,?key,?value):
????????self.__dict__[key]?=?value
????def?__str__(self):
????????return?f"<module?'{self.__name__}'?from?'我來自于虛無'>"
a?=?A("我是?A")
print(a)??#?<module?'我是?A'?from?'我來自于虛無'>
print(a.__name__)??#?我是?A
print(a.xx)??#?不存在的屬性:?xx
a.xx?=?"xx"
print(a.xx)??#?xx
#?加入到?sys.modules?中
sys.modules["嘿嘿"]?=?a
import?嘿嘿
print(嘿嘿.xx)??#?xx
print(嘿嘿.yy)??#?不存在的屬性:?yy
是不是很好玩呢?
小結(jié)
以上就是加載模塊的幾種方式,主要用途如下:
- 導(dǎo)入一個在 sys.path 中的模塊,并且模塊名已知,那么直接使用 import 關(guān)鍵字即可;
- 導(dǎo)入一個在 sys.path 中的模塊,但模塊名是運行時的一個字符串,那么使用 importlib 模塊的 import_module 函數(shù);
- 導(dǎo)入一個不在 sys.path 中的模塊,使用 importlib.machinery 的各種 Loader,只要把模塊的路徑傳進(jìn)去即可。當(dāng)然啦,位于 sys.path 中的模塊也可以使用該方法,但顯然此時使用前兩種更為方便;
- 直接創(chuàng)建一個模塊,通過繼承 module 類來實現(xiàn),并且還可以加入到 sys.modules 中。Python 有一個第三方模塊叫 sh,顧名思義是用來執(zhí)行 Linux Shell 命令的,它內(nèi)部就使用了繼承 module 類來創(chuàng)建模塊的這種方式。但是要知道 module 這個類解釋器沒有暴露給我們,我們需要通過?type(模塊)?或者?模塊.__class__?的方式獲取;
原文鏈接:https://mp.weixin.qq.com/s/q3QqazM46n3hQgQd8EfJJg
相關(guān)推薦
- 2021-12-15 C/C++?Qt?數(shù)據(jù)庫與Chart歷史數(shù)據(jù)展示_C 語言
- 2022-09-21 Android?Intent傳遞大量數(shù)據(jù)出現(xiàn)問題解決_Android
- 2023-01-03 python中import和from-import的區(qū)別解析_python
- 2022-08-21 caffe的python接口生成配置文件學(xué)習(xí)_python
- 2022-02-17 Springboot 獲取響應(yīng)信息,修改響應(yīng)信息
- 2023-02-27 Python獲取"3年前的今天"的日期時間問題_python
- 2022-09-13 c++11中的noexcept關(guān)鍵字_C 語言
- 2022-09-25 Linux基礎(chǔ)組件之死鎖檢測
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支