Python 炫技操作:花式導(dǎo)包的八種方法

直接 import

人盡皆知的方法揪惦,直接導(dǎo)入即可

>>>?import?os

>>>?os.getcwd()

'/home/wangbm'

與此類似的還有误算,不再細(xì)講

import?...

import?...?as?...

from?...?import?...

from?...?import?...?as?...

一般情況下吁脱,使用?import?語句導(dǎo)入模塊已經(jīng)夠用的。

但是在一些特殊場景中谱净,可能還需要其他的導(dǎo)入方式寨蹋。

下面我會一一地給你介紹。

使用 __import__

__import__?函數(shù)可用于導(dǎo)入模塊擂涛,import 語句也會調(diào)用函數(shù)读串。其定義為:

__import__(name[,?globals[,?locals[,?fromlist[,?level]]]])

參數(shù)介紹:

name (required): 被加載 module 的名稱

globals (optional): 包含全局變量的字典,該選項很少使用撒妈,采用默認(rèn)值 global()

locals (optional): 包含局部變量的字典恢暖,內(nèi)部標(biāo)準(zhǔn)實現(xiàn)未用到該變量,采用默認(rèn)值 - local()

fromlist (Optional): 被導(dǎo)入的 submodule 名稱

level (Optional): 導(dǎo)入路徑選項狰右,Python 2 中默認(rèn)為 -1杰捂,表示同時支持 absolute import 和 relative import。Python 3 中默認(rèn)為 0棋蚌,表示僅支持 absolute import嫁佳。如果大于 0,則表示相對導(dǎo)入的父目錄的級數(shù)谷暮,即 1 類似于 '.'蒿往,2 類似于 '..'。

使用示例如下:

>>>?os?=?__import__('os')

>>>?os.getcwd()

'/home/wangbm'

如果要實現(xiàn)?import xx as yy?的效果湿弦,只要修改左值即可

如下示例瓤漏,等價于?import os as myos:

>>>?myos?=?__import__('os')

>>>?myos.getcwd()

'/home/wangbm'

使用 importlib

importlib 是 Python 中的一個標(biāo)準(zhǔn)庫,importlib 能提供的功能非常全面颊埃。

它的簡單示例:

>>>?import?importlib

>>>?myos=importlib.import_module("os")

>>>?myos.getcwd()

'/home/wangbm'

如果要實現(xiàn)?import xx as yy效果蔬充,可以這樣

>>>?import?importlib

>>>?

>>>?myos?=?importlib.import_module("os")

>>>?myos.getcwd()

'/home/wangbm'

使用 imp

imp?模塊提供了一些 import 語句內(nèi)部實現(xiàn)的接口。例如模塊查找(find_module)班利、模塊加載(load_module)等等(模塊的導(dǎo)入過程會包含模塊查找饥漫、加載、緩存等步驟)罗标∮苟樱可以用該模塊來簡單實現(xiàn)內(nèi)建的?__import__?函數(shù)功能:

>>>?import?imp

>>>?file,?pathname,?desc?=?imp.find_module('os')

>>>?myos?=?imp.load_module('sep',?file,?pathname,?desc)

>>>?myos

<module?'sep'?from?'/usr/lib64/python2.7/os.pyc'>

>>>?myos.getcwd()

'/home/wangbm'

從 python 3 開始,內(nèi)建的 reload 函數(shù)被移到了 imp 模塊中馒稍。而從 Python 3.4 開始皿哨,imp 模塊被否決,不再建議使用纽谒,其包含的功能被移到了 importlib 模塊下。即從 Python 3.4 開始如输,importlib 模塊是之前 imp 模塊和 importlib 模塊的合集鼓黔。

使用 execfile

在 Python 2 中有一個 execfile 函數(shù)央勒,利用它可以用來執(zhí)行一個文件。

語法如下:

execfile(filename[,?globals[,?locals]])

參數(shù)有這么幾個:

filename:文件名澳化。

globals:變量作用域崔步,全局命名空間,如果被提供缎谷,則必須是一個字典對象井濒。

locals:變量作用域,局部命名空間列林,如果被提供瑞你,可以是任何映射對象。

>>>?execfile("/usr/lib64/python2.7/os.py")

>>>

>>>?getcwd()

'/home/wangbm'

使用 exec

execfile?只能在 Python2 中使用希痴,Python 3.x 里已經(jīng)刪除了這個函數(shù)者甲。

但是原理值得借鑒,你可以使用 open … read ?讀取文件內(nèi)容砌创,然后再用 exec 去執(zhí)行模塊虏缸。

示例如下:

>>>withopen("/usr/lib64/python2.7/os.py","r")asf:

...?????exec(f.read())

...

>>>?getcwd()

'/home/wangbm'

import_from_github_com

有一個包叫做import_from_github_com,從名字上很容易得知嫩实,它是一個可以從 github 下載安裝并導(dǎo)入的包刽辙。為了使用它,你需要做的就是按照如下命令使用pip 先安裝它甲献。

$?python3?-m?pip?install?import_from_github_com

這個包使用了PEP 302中新的引入鉤子宰缤,允許你可以從github上引入包。這個包實際做的就是安裝這個包并將它添加到本地竟纳。你需要 Python 3.2 或者更高的版本撵溃,并且 git 和 pip 都已經(jīng)安裝才能使用這個包。

pip 要保證是較新版本锥累,如果不是請執(zhí)行如下命令進行升級缘挑。

$?python3?-m?pip?install?--upgrade?pip

確保環(huán)境 ok 后,你就可以在 Python shell 中使用 import_from_github_com

示例如下

>>>fromgithub_com.zzzeekimportsqlalchemy

Collecting?git+https://github.com/zzzeek/sqlalchemy

Cloning?https://github.com/zzzeek/sqlalchemy?to?/tmp/pip-acfv7t06-build

Installing?collected?packages:?SQLAlchemy

Running?setup.py?installforSQLAlchemy?...?done

Successfully?installed?SQLAlchemy-1.1.0b1.dev0

>>>?locals()

{'__builtins__':?,'__spec__':None,

'__package__':None,'__doc__':None,'__name__':'__main__',

'sqlalchemy':?

sqlalchemy/__init__.py'

>,

'__loader__':?}

>>>

看了 import_from_github_com的源碼后桶略,你會注意到它并沒有使用importlib语淘。實際上,它的原理就是使用 pip 來安裝那些沒有安裝的包际歼,然后使用Python的__import__()函數(shù)來引入新安裝的模塊惶翻。

遠程導(dǎo)入模塊

我在這篇文章里(深入探討 Python 的 import 機制:實現(xiàn)遠程導(dǎo)入模塊),深入剖析了導(dǎo)入模塊的內(nèi)部原理鹅心,并在最后手動實現(xiàn)了從遠程服務(wù)器上讀取模塊內(nèi)容吕粗,并在本地成功將模塊導(dǎo)入的導(dǎo)入器。

具體內(nèi)容非常的多旭愧,你可以點擊這個鏈接進行深入學(xué)習(xí)颅筋。

示例代碼如下:

#?新建一個?py?文件(my_importer.py)宙暇,內(nèi)容如下

import?sys

import?importlib

import?urllib.request?as?urllib2

classUrlMetaFinder(importlib.abc.MetaPathFinder):

????def__init__(self,?baseurl):

????????self._baseurl?=?baseurl

????deffind_module(self,?fullname,?path=None):

????????if?path?is?None:

????????????baseurl?=?self._baseurl

????????else:

????????????#?不是原定義的url就直接返回不存在

????????????if?not?path.startswith(self._baseurl):

????????????????return?None

????????????baseurl?=?path

????????try:

????????????loader?=?UrlMetaLoader(baseurl)

??return?loader

????????except?Exception:

????????????return?None

classUrlMetaLoader(importlib.abc.SourceLoader):

????def__init__(self,?baseurl):

????????self.baseurl?=?baseurl

????defget_code(self,?fullname):

????????f?=?urllib2.urlopen(self.get_filename(fullname))

????????return?f.read()

????defget_data(self):

????????pass

????defget_filename(self,?fullname):

????????return?self.baseurl?+?fullname?+?'.py'

definstall_meta(address):

????finder?=?UrlMetaFinder(address)

????sys.meta_path.append(finder)

并且在遠程服務(wù)器上開啟 http 服務(wù)(為了方便,我僅在本地進行演示)议泵,并且手動編輯一個名為 my_info 的 python 文件占贫,如果后面導(dǎo)入成功會打印?ok。

$mkdir?httpserver?&&cdhttpserver

$?cat>my_info.py<EOF

name='wangbm'

print('ok')

EOF

$?cat?my_info.py

name='wangbm'

print('ok')

$

$?python3?-m?http.server?12800

Serving?HTTP?on?0.0.0.0?port?12800?(http://0.0.0.0:12800/)?...

...

一切準(zhǔn)備好先口,驗證開始型奥。

>>>?from?my_importer?import?install_meta

>>>?install_meta('http://localhost:12800/')?#?往?sys.meta_path?注冊?finder

>>>?import?my_info??#?打印ok,說明導(dǎo)入成功

ok

>>>?my_info.name??#?驗證可以取得到變量

'wangbm'

好了碉京,8 種方法都給大家介紹完畢厢汹,對于普通開發(fā)者來說,其實只要掌握 import 這種方法足夠了收夸,而對于那些想要自己開發(fā)框架的人來說坑匠,深入學(xué)習(xí) __import__?以及 importlib 是非常有必要的。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末卧惜,一起剝皮案震驚了整個濱河市厘灼,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌咽瓷,老刑警劉巖设凹,帶你破解...
    沈念sama閱讀 219,188評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異茅姜,居然都是意外死亡闪朱,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,464評論 3 395
  • 文/潘曉璐 我一進店門钻洒,熙熙樓的掌柜王于貴愁眉苦臉地迎上來奋姿,“玉大人,你說我怎么就攤上這事素标〕剖” “怎么了?”我有些...
    開封第一講書人閱讀 165,562評論 0 356
  • 文/不壞的土叔 我叫張陵头遭,是天一觀的道長寓免。 經(jīng)常有香客問我,道長计维,這世上最難降的妖魔是什么袜香? 我笑而不...
    開封第一講書人閱讀 58,893評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮鲫惶,結(jié)果婚禮上蜈首,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好疾就,可當(dāng)我...
    茶點故事閱讀 67,917評論 6 392
  • 文/花漫 我一把揭開白布澜术。 她就那樣靜靜地躺著艺蝴,像睡著了一般猬腰。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上猜敢,一...
    開封第一講書人閱讀 51,708評論 1 305
  • 那天姑荷,我揣著相機與錄音,去河邊找鬼缩擂。 笑死鼠冕,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的胯盯。 我是一名探鬼主播懈费,決...
    沈念sama閱讀 40,430評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼博脑!你這毒婦竟也來了憎乙?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,342評論 0 276
  • 序言:老撾萬榮一對情侶失蹤叉趣,失蹤者是張志新(化名)和其女友劉穎泞边,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體疗杉,經(jīng)...
    沈念sama閱讀 45,801評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡阵谚,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,976評論 3 337
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了烟具。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片梢什。...
    茶點故事閱讀 40,115評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖朝聋,靈堂內(nèi)的尸體忽然破棺而出嗡午,到底是詐尸還是另有隱情,我是刑警寧澤玖翅,帶...
    沈念sama閱讀 35,804評論 5 346
  • 正文 年R本政府宣布翼馆,位于F島的核電站,受9級特大地震影響金度,放射性物質(zhì)發(fā)生泄漏应媚。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,458評論 3 331
  • 文/蒙蒙 一猜极、第九天 我趴在偏房一處隱蔽的房頂上張望中姜。 院中可真熱鬧,春花似錦、人聲如沸丢胚。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,008評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽携龟。三九已至兔跌,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間峡蟋,已是汗流浹背坟桅。 一陣腳步聲響...
    開封第一講書人閱讀 33,135評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留蕊蝗,地道東北人仅乓。 一個月前我還...
    沈念sama閱讀 48,365評論 3 373
  • 正文 我出身青樓,卻偏偏與公主長得像蓬戚,于是被迫代替她去往敵國和親夸楣。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,055評論 2 355

推薦閱讀更多精彩內(nèi)容