[譯]The Python Tutorial#Brief Tour of the Standard Library — Part II

[譯]The Python Tutorial#Brief Tour of the Standard Library — Part II

第二部分介紹更多滿足專業(yè)編程需求的高級模塊,這些模塊在小型腳本中很少用到缰泡。

11.1 Output Formatting

reprlib模塊為大型或者深度嵌套的容器提供了一個定制版本的repr()函數(shù):

>>> import reprlib
>>> reprlib.repr(set('supercalifragilisticexpialidocious'))
"{'a', 'c', 'd', 'e', 'f', 'g', ...}"

pprint模塊為內(nèi)嵌對象或者用戶自定義對象以解釋器可讀方式打印提供了更精細(xì)的控制词裤。當(dāng)打印結(jié)果超過一行時相满,“打印美化器”添加換行符和縮進菜枷,清晰顯示原有的數(shù)據(jù)結(jié)構(gòu):

>>> import pprint
>>> t = [[[['black', 'cyan'], 'white', ['green', 'red']], [['magenta',
...     'yellow'], 'blue']]]
...
>>> pprint.pprint(t, width=30)
[[[['black', 'cyan'],
   'white',
   ['green', 'red']],
  [['magenta', 'yellow'],
   'blue']]]

textwrap模塊格式化文本段落適應(yīng)指定屏幕寬度:

>>> import textwrap
>>> doc = """The wrap() method is just like fill() except that it returns
... a list of strings instead of one big string with newlines to separate
... the wrapped lines."""
...
>>> print(textwrap.fill(doc, width=40))
The wrap() method is just like fill()
except that it returns a list of strings
instead of one big string with newlines
to separate the wrapped lines.

local模塊訪問特定文化數(shù)據(jù)格式的數(shù)據(jù)庫。區(qū)域設(shè)置格式函數(shù)的分組屬性提供了使用組分隔符格式化數(shù)字的直接方法:

>>> import locale
>>> locale.setlocale(locale.LC_ALL, 'English_United States.1252')
'English_United States.1252'
>>> conv = locale.localeconv()          # get a mapping of conventions
>>> x = 1234567.8
>>> locale.format("%d", x, grouping=True)
'1,234,567'
>>> locale.format_string("%s%.*f", (conv['currency_symbol'],
...                      conv['frac_digits'], x), grouping=True)
'$1,234,567.80'

11.2 Templting

string模塊包括一個功能強大的Template類吭练,其簡化的語法適用于最終用戶的編輯冕茅。該類允許用戶自定義他們的應(yīng)用,而不用修改應(yīng)用店茶。

格式化使用由$和有效Python標(biāo)識符(字母數(shù)字字符以及下劃線)組成的占位符蜕便。占位符外圍的花括號允許其后跟更多數(shù)字字母字符而無需中間空格。使用$$轉(zhuǎn)義$

>>> from string import Template
>>> t = Template('${village}folk send $$10 to $cause.')
>>> t.substitute(village='Nottingham', cause='the ditch fund')
'Nottinghamfolk send $10 to the ditch fund.'

當(dāng)參數(shù)字典或者關(guān)鍵字參數(shù)沒有提供對應(yīng)的占位符時贩幻,substitute()方法拋出KeyError異常轿腺。對于郵件合并風(fēng)格的應(yīng)用程序两嘴,用戶提供的數(shù)據(jù)可能不完全,使用safe_substitute()方法更加合適——該方法會保留為匹配的占位符不變:

>>> t = Template('Return the $item to $owner.')
>>> d = dict(item='unladen swallow')
>>> t.substitute(d)
Traceback (most recent call last):
  ...
KeyError: 'owner'
>>> t.safe_substitute(d)
'Return the unladen swallow to $owner.'

Template的子類可以指定自定義的定界符族壳。例如憔辫,圖片瀏覽器的批量重命名工具可能使用百分比號作為占位符,如當(dāng)前日期仿荆,圖片序列碼或者文件格式:

>>> import time, os.path
>>> photofiles = ['img_1074.jpg', 'img_1076.jpg', 'img_1077.jpg']
>>> class BatchRename(Template):
...     delimiter = '%'
>>> fmt = input('Enter rename style (%d-date %n-seqnum %f-format):  ')
Enter rename style (%d-date %n-seqnum %f-format):  Ashley_%n%f

>>> t = BatchRename(fmt)
>>> date = time.strftime('%d%b%y')
>>> for i, filename in enumerate(photofiles):
...     base, ext = os.path.splitext(filename)
...     newname = t.substitute(d=date, n=i, f=ext)
...     print('{0} --> {1}'.format(filename, newname))

img_1074.jpg --> Ashley_0.jpg
img_1076.jpg --> Ashley_1.jpg
img_1077.jpg --> Ashley_2.jpg

模板的另一種應(yīng)用是將程序邏輯與多種輸出格式的細(xì)節(jié)分開贰您。這樣使得自定義模板替換為XML文件,純文本報告和HTML網(wǎng)絡(luò)報告成為可能赖歌。

11.3 Working with Binary Data Record Layouts

struct模塊提供了用于處理可變長度二進制記錄格式的pack以及unpack枉圃。以下示例展示如何在沒有zipfile的幫助下遍歷ZIP文件的頭信息。分組代碼“H”和“I”分別表示兩個和四個字節(jié)的無符號數(shù)庐冯。<表示它們是標(biāo)準(zhǔn)大小和小端字節(jié)順序:

import struct

with open('myfile.zip', 'rb') as f:
    data = f.read()

start = 0
for i in range(3):                      # show the first 3 file headers
    start += 14
    fields = struct.unpack('<IIIHH', data[start:start+16])
    crc32, comp_size, uncomp_size, filenamesize, extra_size = fields

    start += 16
    filename = data[start:start+filenamesize]
    start += filenamesize
    extra = data[start:start+extra_size]
    print(filename, hex(crc32), comp_size, uncomp_size)

    start += extra_size + comp_size     # skip to the next header

11.4 Multi-threading

線程是一種解耦無順序依賴關(guān)系任務(wù)的技術(shù)孽亲。線程有助于提高接收用戶輸入應(yīng)用的響應(yīng)速度,而其他任務(wù)在后臺運行展父。相關(guān)的用例是運行IO的同時在另一個線程中作耗時計算操作返劲。

以下代碼展示高層模塊threading如何在主程序運行的同時在后臺執(zhí)行任務(wù):

import threading, zipfile

class AsyncZip(threading.Thread):
    def __init__(self, infile, outfile):
        threading.Thread.__init__(self)
        self.infile = infile
        self.outfile = outfile

    def run(self):
        f = zipfile.ZipFile(self.outfile, 'w', zipfile.ZIP_DEFLATED)
        f.write(self.infile)
        f.close()
        print('Finished background zip of:', self.infile)

background = AsyncZip('mydata.txt', 'myarchive.zip')
background.start()
print('The main program continues to run in foreground.')

background.join()    # Wait for the background task to finish
print('Main program waited until background was done.')

多線程應(yīng)用主要的挑戰(zhàn)是協(xié)調(diào)共享數(shù)據(jù)或者其他資源的多個線程。為了實現(xiàn)這個目標(biāo)栖茉,線程模塊提供了一系列同步原語篮绿,包括鎖,時間吕漂,條件變量以及信號亲配。

雖然這些工具很強大,但是很小的設(shè)計錯誤也會導(dǎo)致很難重現(xiàn)的問題惶凝。因此吼虎,任務(wù)協(xié)作首選的方案是將所有對資源的訪問都集中在一個單線程中,然后使用queue模塊為這個單線程提供來自其他線程的請求苍鲜。使用Queue對象進行跨線程通信和協(xié)調(diào)的應(yīng)用程序更容易設(shè)計思灰,可讀性更好,更可靠混滔。

11.5 Logging

logging模塊提供了功能完備且靈活的日志系統(tǒng)洒疚。最簡單的用法是將日志信息輸出到文件或者sys.stderr

import logging
logging.debug('Debugging information')
logging.info('Informational message')
logging.warning('Warning:config file %s not found', 'server.conf')
logging.error('Error occurred')
logging.critical('Critical error -- shutting down')

輸出如下:

WARNING:root:Warning:config file server.conf not found
ERROR:root:Error occurred
CRITICAL:root:Critical error -- shutting down

默認(rèn)情況下,debug和info級別的信息被抑制坯屿,輸出目的地是標(biāo)準(zhǔn)錯誤油湖。其他輸出選項包括通過郵件,數(shù)據(jù)報领跛,套接字以及HTTP服務(wù)器的路由信息肺魁。新的過濾器可以根據(jù)不同的消息優(yōu)先級選擇不同的路由:DEBUG, INFO, WARNING, ERROR以及CRITICAL

日志系統(tǒng)可以直接從Python配置隔节,也可以從用戶可編輯的配置文件加載鹅经,以便于自定義日志,而無需修改應(yīng)用程序怎诫。

11.6 Weak References

Python自動進行內(nèi)存管理(針對大多數(shù)對象引用計數(shù)以及垃圾回收以消除循環(huán))瘾晃。對象最后的引用刪除后,其內(nèi)存立即釋放幻妓。

這種方法適用于大多數(shù)應(yīng)用蹦误,但是偶爾只需要跟蹤對象即可。不幸的是肉津,僅僅跟蹤它們也會創(chuàng)建永久的應(yīng)用强胰。weakref模塊提供了無需創(chuàng)建引用跟蹤對象的工具。一旦不再需要對象時妹沙,其自動從弱引用表中刪除偶洋,并觸發(fā)回調(diào)。典型的應(yīng)用包括創(chuàng)建成本高的緩存對象:

>>> import weakref, gc
>>> class A:
...     def __init__(self, value):
...         self.value = value
...     def __repr__(self):
...         return str(self.value)
...
>>> a = A(10)                   # create a reference
>>> d = weakref.WeakValueDictionary()
>>> d['primary'] = a            # does not create a reference
>>> d['primary']                # fetch the object if it is still alive
10
>>> del a                       # remove the one reference
>>> gc.collect()                # run garbage collection right away
0
>>> d['primary']                # entry was automatically removed
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
    d['primary']                # entry was automatically removed
  File "C:/python36/lib/weakref.py", line 46, in __getitem__
    o = self.data[key]()
KeyError: 'primary'

11.7 Tools for Working with Lists

內(nèi)置列表類型可以滿足許多數(shù)據(jù)結(jié)構(gòu)的需求距糖。然而玄窝,有時候需要不同性能權(quán)衡的替代實現(xiàn)。

array模塊提供一個array()對象悍引,就像一個只存儲同類型數(shù)據(jù)恩脂,并且存儲更加緊湊的列表一樣。以下示例展示一個數(shù)字?jǐn)?shù)組趣斤,它以兩個字節(jié)無符號二進制數(shù)(typecode為"H")存儲俩块,而不是通常的每個對象16字節(jié)的常規(guī)Python整數(shù)對象列表:

>>> from array import array
>>> a = array('H', [4000, 10, 700, 22222])
>>> sum(a)
26932
>>> a[1:3]
array('H', [10, 700])

collections模塊提供deque()對象,就像一個擁有速度更快的左端添加和移除操作浓领,更慢的中間查找速度的列表一樣玉凯。這些對象適用于實現(xiàn)隊列以及寬度優(yōu)先樹搜索:

>>> from collections import deque
>>> d = deque(["task1", "task2", "task3"])
>>> d.append("task4")
>>> print("Handling", d.popleft())
Handling task1
unsearched = deque([starting_node])
def breadth_first_search(unsearched):
    node = unsearched.popleft()
    for m in gen_moves(node):
        if is_goal(m):
            return m
        unsearched.append(m)

除了可選的列表實現(xiàn)外,庫也提供了其他工具镊逝,比如具有操作有序列表函數(shù)的bisect模塊:

>>> import bisect
>>> scores = [(100, 'perl'), (200, 'tcl'), (400, 'lua'), (500, 'python')]
>>> bisect.insort(scores, (300, 'ruby'))
>>> scores
[(100, 'perl'), (200, 'tcl'), (300, 'ruby'), (400, 'lua'), (500, 'python')]

heapq模塊提供了基于常規(guī)隊列實現(xiàn)堆的函數(shù)壮啊。最小值的實體總是放在位置0處。這對于需要重復(fù)訪問最小元素但是不想要對整個列表排序的應(yīng)用很有用:

>>> from heapq import heapify, heappop, heappush
>>> data = [1, 3, 5, 7, 9, 2, 4, 6, 8, 0]
>>> heapify(data)                      # rearrange the list into heap order
>>> heappush(data, -5)                 # add a new entry
>>> [heappop(data) for i in range(3)]  # fetch the three smallest entries
[-5, 0, 1]

11.8 Decimal Floating Point Arithmetic

decimal模塊為十進制浮點算法提供Decimal數(shù)據(jù)類型撑蒜。與二進制浮點型的內(nèi)嵌的float實現(xiàn)相比歹啼,這個類在以下場合及其有用:

  • 金融類應(yīng)用以及其他需要精確十進制表示的應(yīng)用;
  • 控制精度座菠;
  • 控制舍入以滿足法律上或者管控上的需求狸眼;
  • 追蹤重要的小數(shù)位數(shù),或者
  • 用戶其他輸出與手工計算匹配的應(yīng)用

例如浴滴,計算對70分的電話費取5%的稅拓萌,十進制浮點型盒二進制浮點型結(jié)果不同。當(dāng)結(jié)果需要舍入到最近的整分時升略,它們之間的差異就很重要了:

>>> from decimal import *
>>> round(Decimal('0.70') * Decimal('1.05'), 2)
Decimal('0.74')
>>> round(.70 * 1.05, 2)
0.73

十進制結(jié)果保持一個尾隨的0微王,精度自從從兩位延伸到了四位屡限。十進制可以重新進行數(shù)學(xué)運算,就像手工一樣炕倘,并且避免了二進制浮點數(shù)不能精確表示十進制數(shù)時會產(chǎn)生的問題钧大。

精確表示使得Decimal類可以執(zhí)行求余計算以及不適用于二進制浮點型的比較測試:

>>> Decimal('1.00') % Decimal('.10')
Decimal('0.00')
>>> 1.00 % 0.10
0.09999999999999995

>>> sum([Decimal('0.1')]*10) == Decimal('1.0')
True
>>> sum([0.1]*10) == 1.0
False

decimal模塊提供了必要的高精度算法:

>>> getcontext().prec = 36
>>> Decimal(1) / Decimal(7)
Decimal('0.142857142857142857142857142857142857')
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市罩旋,隨后出現(xiàn)的幾起案子啊央,更是在濱河造成了極大的恐慌,老刑警劉巖涨醋,帶你破解...
    沈念sama閱讀 219,039評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件瓜饥,死亡現(xiàn)場離奇詭異,居然都是意外死亡浴骂,警方通過查閱死者的電腦和手機乓土,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,426評論 3 395
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來靠闭,“玉大人帐我,你說我怎么就攤上這事±颍” “怎么了拦键?”我有些...
    開封第一講書人閱讀 165,417評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長檩淋。 經(jīng)常有香客問我芬为,道長,這世上最難降的妖魔是什么蟀悦? 我笑而不...
    開封第一講書人閱讀 58,868評論 1 295
  • 正文 為了忘掉前任媚朦,我火速辦了婚禮,結(jié)果婚禮上日戈,老公的妹妹穿的比我還像新娘询张。我一直安慰自己,他們只是感情好浙炼,可當(dāng)我...
    茶點故事閱讀 67,892評論 6 392
  • 文/花漫 我一把揭開白布份氧。 她就那樣靜靜地躺著,像睡著了一般弯屈。 火紅的嫁衣襯著肌膚如雪蜗帜。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,692評論 1 305
  • 那天资厉,我揣著相機與錄音厅缺,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛湘捎,可吹牛的內(nèi)容都是我干的诀豁。 我是一名探鬼主播,決...
    沈念sama閱讀 40,416評論 3 419
  • 文/蒼蘭香墨 我猛地睜開眼消痛,長吁一口氣:“原來是場噩夢啊……” “哼且叁!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起秩伞,我...
    開封第一講書人閱讀 39,326評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎欺矫,沒想到半個月后纱新,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,782評論 1 316
  • 正文 獨居荒郊野嶺守林人離奇死亡穆趴,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,957評論 3 337
  • 正文 我和宋清朗相戀三年脸爱,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片未妹。...
    茶點故事閱讀 40,102評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡簿废,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出络它,到底是詐尸還是另有隱情族檬,我是刑警寧澤,帶...
    沈念sama閱讀 35,790評論 5 346
  • 正文 年R本政府宣布化戳,位于F島的核電站单料,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏点楼。R本人自食惡果不足惜扫尖,卻給世界環(huán)境...
    茶點故事閱讀 41,442評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望掠廓。 院中可真熱鬧换怖,春花似錦、人聲如沸蟀瞧。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,996評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽黄橘。三九已至兆览,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間塞关,已是汗流浹背抬探。 一陣腳步聲響...
    開封第一講書人閱讀 33,113評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人小压。 一個月前我還...
    沈念sama閱讀 48,332評論 3 373
  • 正文 我出身青樓线梗,卻偏偏與公主長得像,于是被迫代替她去往敵國和親怠益。 傳聞我的和親對象是個殘疾皇子仪搔,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,044評論 2 355

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

  • 國家電網(wǎng)公司企業(yè)標(biāo)準(zhǔn)(Q/GDW)- 面向?qū)ο蟮挠秒娦畔?shù)據(jù)交換協(xié)議 - 報批稿:20170802 前言: 排版 ...
    庭說閱讀 10,986評論 6 13
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)蜻牢,斷路器烤咧,智...
    卡卡羅2017閱讀 134,661評論 18 139
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法,類相關(guān)的語法抢呆,內(nèi)部類的語法煮嫌,繼承相關(guān)的語法,異常的語法抱虐,線程的語...
    子非魚_t_閱讀 31,639評論 18 399
  • 我遇見你昌阿,是最美麗的意外。 Chapter one 她叫狗小姐恳邀,但這并不奇怪懦冰。狗小姐由爺爺奶奶養(yǎng)大,自小在鄉(xiāng)間野慣...
    三石妹妹閱讀 318評論 14 8
  • 活動單位:唯品網(wǎng)球俱樂部 活動時間:45分鐘 流程: 1.糖老師 介紹網(wǎng)球歷史及好處 5分鐘 2.回顧教學(xué) 糖...
    知了唯知愛閱讀 291評論 0 0