恕我直言饼拍!你對(duì)Python里的import一無所知

寫 Python 通常我們會(huì)怎樣導(dǎo)包赡模?可能大部分情況下都是用 import,但除了 import 你還會(huì)些什么呢惕耕?下面我們來介紹一些騷操作纺裁。

  1. 直接 import

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

>>> import os
>>> os.getcwd()
'/home/wangbm'

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

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

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

但是在一些特殊場(chǎng)景中挤安,可能還需要其他的導(dǎo)入方式谚殊。

下面我會(huì)一一地給你介紹。

  1. 使用 import

import 函數(shù)可用于導(dǎo)入模塊蛤铜,import 語句也會(huì)調(diào)用函數(shù)嫩絮。其定義為:

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

參數(shù)介紹:

name (required): 被加載 module 的名稱
globals (optional): 包含全局變量的字典丛肢,該選項(xiàng)很少使用,采用默認(rèn)值 global()
locals (optional): 包含局部變量的字典剿干,內(nèi)部標(biāo)準(zhǔn)實(shí)現(xiàn)未用到該變量蜂怎,采用默認(rèn)值 - local()
fromlist (Optional): 被導(dǎo)入的 submodule 名稱
level (Optional): 導(dǎo)入路徑選項(xiàng),Python 2 中默認(rèn)為 -1置尔,表示同時(shí)支持 absolute import 和 relative import杠步。Python 3 中默認(rèn)為 0,表示僅支持 absolute import榜轿。如果大于 0幽歼,則表示相對(duì)導(dǎo)入的父目錄的級(jí)數(shù),即 1 類似于 '.'谬盐,2 類似于 '..'甸私。
使用示例如下:

>>> os = __import__('os')
>>> os.getcwd()
'/home/wangbm'

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

如下示例飞傀,等價(jià)于 import os as myos:

>>> myos = __import__('os')
>>> myos.getcwd()
'/home/wangbm'
  1. 使用 importlib

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

它的簡(jiǎn)單示例:

>>> import importlib
>>> myos=importlib.import_module("os")
>>> myos.getcwd()
'/home/wangbm'

如果要實(shí)現(xiàn) import xx as yy效果助析,可以這樣

>>> import importlib
>>> 
>>> myos = importlib.import_module("os")
>>> myos.getcwd()
'/home/wangbm'
  1. 使用 imp

imp 模塊提供了一些 import 語句內(nèi)部實(shí)現(xiàn)的接口犀被。例如模塊查找(find_module)、模塊加載(load_module)等等(模塊的導(dǎo)入過程會(huì)包含模塊查找外冀、加載寡键、緩存等步驟)⊙┧恚可以用該模塊來簡(jiǎn)單實(shí)現(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 模塊的合集。

  1. 使用 execfile

在 Python 2 中有一個(gè) execfile 函數(shù)措近,利用它可以用來執(zhí)行一個(gè)文件溶弟。

語法如下:

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

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

filename:文件名。
globals:變量作用域瞭郑,全局命名空間辜御,如果被提供,則必須是一個(gè)字典對(duì)象屈张。
locals:變量作用域擒权,局部命名空間袱巨,如果被提供,可以是任何映射對(duì)象碳抄。

>>> execfile("/usr/lib64/python2.7/os.py")
>>> 
>>> getcwd()
'/home/wangbm'
  1. 使用 exec

execfile 只能在 Python2 中使用愉老,Python 3.x 里已經(jīng)刪除了這個(gè)函數(shù)。

但是原理值得借鑒纳鼎,你可以使用 open … read 讀取文件內(nèi)容俺夕,然后再用 exec 去執(zhí)行模塊裳凸。

示例如下:

>>> with open("/usr/lib64/python2.7/os.py", "r") as f:
...     exec(f.read())
... 
>>> getcwd()
'/home/wangbm'
  1. import_from_github_com

有一個(gè)包叫做 import_from_github_com贱鄙,從名字上很容易得知,它是一個(gè)可以從 github 下載安裝并導(dǎo)入的包姨谷。為了使用它逗宁,你需要做的就是按照如下命令使用pip 先安裝它。

$ python3 -m pip install import_from_github_com

這個(gè)包使用了PEP 302中新的引入鉤子梦湘,允許你可以從github上引入包瞎颗。這個(gè)包實(shí)際做的就是安裝這個(gè)包并將它添加到本地。你需要 Python 3.2 或者更高的版本捌议,并且 git 和 pip 都已經(jīng)安裝才能使用這個(gè)包哼拔。

pip 要保證是較新版本,如果不是請(qǐng)執(zhí)行如下命令進(jìn)行升級(jí)瓣颅。

$ python3 -m pip install --upgrade pip

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

示例如下

>>> from github_com.zzzeek import sqlalchemy
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 install for SQLAlchemy ... done
Successfully installed SQLAlchemy-1.1.0b1.dev0
>>> locals()
{'__builtins__': <module 'builtins' (built-in)>, '__spec__': None,
'__package__': None, '__doc__': None, '__name__': '__main__',
'sqlalchemy': <module 'sqlalchemy' from '/usr/local/lib/python3.5/site-packages/\
sqlalchemy/__init__.py'>,
'__loader__': <class '_frozen_importlib.BuiltinImporter'>}
>>>

看了 import_from_github_com的源碼后,你會(huì)注意到它并沒有使用importlib宫补。實(shí)際上檬姥,它的原理就是使用 pip 來安裝那些沒有安裝的包,然后使用Python的__import__()函數(shù)來引入新安裝的模塊粉怕。

8. 遠(yuǎn)程導(dǎo)入模塊

示例代碼如下:

# 新建一個(gè) py 文件(my_importer.py)健民,內(nèi)容如下
import sys
import importlib
import urllib.request as urllib2

class UrlMetaFinder(importlib.abc.MetaPathFinder):
    def __init__(self, baseurl):
        self._baseurl = baseurl


    def find_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

class UrlMetaLoader(importlib.abc.SourceLoader):
    def __init__(self, baseurl):
        self.baseurl = baseurl

    def get_code(self, fullname):
        f = urllib2.urlopen(self.get_filename(fullname))
        return f.read()

    def get_data(self):
        pass

    def get_filename(self, fullname):
        return self.baseurl + fullname + '.py'

def install_meta(address):
    finder = UrlMetaFinder(address)
    sys.meta_path.append(finder)

并且在遠(yuǎn)程服務(wù)器上開啟 http 服務(wù)(為了方便,我僅在本地進(jìn)行演示)贫贝,并且手動(dòng)編輯一個(gè)名為 my_info 的 python 文件秉犹,如果后面導(dǎo)入成功會(huì)打印 ok。

$ mkdir httpserver && cd httpserver
$ 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)備好稚晚,驗(yàn)證開始崇堵。

>>> from my_importer import install_meta
>>> install_meta('http://localhost:12800/') # 往 sys.meta_path 注冊(cè) finder
>>> import my_info  # 打印ok,說明導(dǎo)入成功
ok
>>> my_info.name  # 驗(yàn)證可以取得到變量
'wangbm'

好了蜈彼,8 種方法都給大家介紹完畢筑辨,對(duì)于普通開發(fā)者來說,其實(shí)只要掌握 import 這種方法足夠了幸逆,而對(duì)于那些想要自己開發(fā)框架的人來說棍辕,深入學(xué)習(xí) import 以及 importlib 是非常有必要的暮现。

工具神器大集合:

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末栖袋,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子抚太,更是在濱河造成了極大的恐慌塘幅,老刑警劉巖,帶你破解...
    沈念sama閱讀 207,113評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件尿贫,死亡現(xiàn)場(chǎng)離奇詭異电媳,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)庆亡,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評(píng)論 2 381
  • 文/潘曉璐 我一進(jìn)店門匾乓,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人又谋,你說我怎么就攤上這事拼缝。” “怎么了彰亥?”我有些...
    開封第一講書人閱讀 153,340評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵咧七,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我任斋,道長(zhǎng)继阻,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,449評(píng)論 1 279
  • 正文 為了忘掉前任仁卷,我火速辦了婚禮穴翩,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘锦积。我一直安慰自己芒帕,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,445評(píng)論 5 374
  • 文/花漫 我一把揭開白布丰介。 她就那樣靜靜地躺著背蟆,像睡著了一般。 火紅的嫁衣襯著肌膚如雪哮幢。 梳的紋絲不亂的頭發(fā)上带膀,一...
    開封第一講書人閱讀 49,166評(píng)論 1 284
  • 那天,我揣著相機(jī)與錄音橙垢,去河邊找鬼垛叨。 笑死,一個(gè)胖子當(dāng)著我的面吹牛柜某,可吹牛的內(nèi)容都是我干的嗽元。 我是一名探鬼主播敛纲,決...
    沈念sama閱讀 38,442評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼剂癌!你這毒婦竟也來了淤翔?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,105評(píng)論 0 261
  • 序言:老撾萬榮一對(duì)情侶失蹤佩谷,失蹤者是張志新(化名)和其女友劉穎旁壮,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體谐檀,經(jīng)...
    沈念sama閱讀 43,601評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡抡谐,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,066評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了稚补。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片童叠。...
    茶點(diǎn)故事閱讀 38,161評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖课幕,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情五垮,我是刑警寧澤乍惊,帶...
    沈念sama閱讀 33,792評(píng)論 4 323
  • 正文 年R本政府宣布,位于F島的核電站放仗,受9級(jí)特大地震影響润绎,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜诞挨,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,351評(píng)論 3 307
  • 文/蒙蒙 一莉撇、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧惶傻,春花似錦棍郎、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至蜈敢,卻和暖如春辜荠,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背抓狭。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評(píng)論 1 261
  • 我被黑心中介騙來泰國(guó)打工伯病, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人否过。 一個(gè)月前我還...
    沈念sama閱讀 45,618評(píng)論 2 355
  • 正文 我出身青樓午笛,卻偏偏與公主長(zhǎng)得像膨蛮,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子季研,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,916評(píng)論 2 344