1. 程式人生 > 實用技巧 >6. sys庫

6. sys庫

sys模組主要是針對與Python直譯器相關的變數和方法,不是主機作業系統。

匯入方式:import sys

1. sys.argv

sys.argv是一個指令碼執行引數列表,列表的第一個元素是指令碼名稱,從第二個元素開始才是真正的引數。

#test.py

import sys

for index,arg in enumerate(sys.argv):

print("第%d個引數是: %s"%(index,arg))

執行python test.py 1 2 3 4,結果:

第0個引數是: test.py

第1個引數是: 1

第2個引數是: 2

第3個引數是: 3

第4個引數是: 4

2. sys.getrefcount(object)

我們都知道Python有自動的垃圾回收機制,讓我們不用費力去進行記憶體管理。那麼Python怎麼知道一個物件可以被當做垃圾回收呢?Python使用‘引用計數’的方式,追蹤每個物件 的引用次數,每對這個物件的一次引用,這個計數就加一,每刪除一個該物件的引用,這個計數就減一。當引用為0的時候,就表示沒有任何變數指向這個物件,那麼就可以回收這個物件,騰出它所佔用的記憶體空間。

sys.getrefcount(object)這個方法可以返回一個物件被引用的次數。注意,這個次數預設從1開始,因為你在使用sys.getrefcount(object)方法的時候就已經引用了它一次(該引用是臨時性的,呼叫結束後,自動解除引用。)。如果不好理解,可以簡單地認為它自帶被動光環:引用+1。

>>> a = "I like Python!"

>>> sys.getrefcount(a)

2

>>> b = a

>>> sys.getrefcount(a)

3

>>> c = a

>>> sys.getrefcount(a)

4

>>> del c

>>> sys.getrefcount(a)

3

>>> del b

>>> sys.getrefcount(a)

2

>>> sys.getrefcount(1)

902

>>> sys.getrefcount("a")

36

>>> sys.getrefcount(True)

581

>>> sys.getrefcount(None)

6918

注意例項中的1、"a"、True、None,Python內部環境執行過程中已經引用了它們很多次,None甚至被使用了6918次。

3. sys.modules

sys.modules儲存有當前Python環境中已經匯入的模組記錄,這是一個全域性字典,當Python啟動後就載入在記憶體中。每當匯入新的模組,sys.modules將自動記錄該模組,當第二次試圖再次匯入該模組時,Python會先到這個字典中查詢是否曾經匯入過該模組。是則忽略,否則匯入,從而加快了程式執行的速度。同時,它擁有字典的基本方法。例如sys.modules.keys()檢視字典的所有鍵,sys.modules.values()檢視字典的所有值,sys.modules['sys']檢視sys鍵對應的值。

>>>importsys

>>>sys.modules

{'builtins':<module 'builtins' (built-in)>,'sys':<module 'sys' (built-in)>,'_frozen_importlib':<module 'importlib._bootstrap' (frozen)>, '_imp' :<module'_imp'(built-in)>, '_warnings':<module'_warnings'(built-in)>,'_thread':.......擷取部分

>>>sys.modules.keys()

dict_keys(['builtins','sys', '_frozen_importlib', '_imp','_warnings', '_thread','_weakref', '_frozen_importlib_external', '_io','marshal','nt','winreg', 'zipimport','encodings', 'codecs', '_codecs', 'encodings.aliases', ...擷取部分

>>>sys.modules.values()

dict_values([<module 'builtins' (built-in)>,<module 'sys' (built-in)>,<module 'importlib._bootstrap' (frozen)>,<module '_imp' (built-in)>,<module '_warnings' (built-in)>,<module '_thread' (built-in)>,<module '_weakref' module' urllib.parse' from' C:\\Python36\\lib\\urllib\\parse.py'>]).......擷取部分

>>>sys.modules['sys']

<module'sys'(built-in)>

4. sys.builtin_module_names

sys.builtin_module_names是一個字串元組,包含了所有已經編譯在Python直譯器內的模組名稱。

import sys

def find_module(module):

if module in sys.builtin_module_names:

print(module," 內置於=> ","__builtin__")

else:

print(module,"模組位於=> ",__import__(module).__file__)

find_module('os')

find_module('sys')

find_module('time')

find_module('zlib')

find_module('string')

#----------

執行結果:

os模組位於=>C:\Python36\lib\os.py

sys內置於=>__builtin__

time內置於=>__builtin__

zlib內置於=>__builtin__

string模組位於=>C:\Python36\lib\string.py

5. sys.path

path是一個目錄列表,供Python從中查詢模組。在Python啟動時,sys.path根據內建規則和PYTHONPATH變數進行初始化。sys.path的第一個元素通常是個空字串,表示當前目錄。

>>> sys.path

['', 'C:\\Python36\\Lib\\idlelib', 'C:\\Python36\\python36.zip', 'C:\\Python36\\DLLs', 'C:\\Python36\\lib', 'C:\\Python36', 'C:\\Python36\\lib\\site-packages']

sys.path本質上是一個列表,可以進行append、insert、pop、remove等各種列表相關的操作,但通常都進行append操作,新增自己想要的查詢路徑。在做修改、刪除型別的操作之前,請務必確認你的行為!

6. sys.platform

獲取當前執行環境的平臺名稱,不同的平臺返回值如下表所示:

7. sys.stdin、sys.stdout、sys.stderr

stdin用於所有的互動式輸入(包括input()函式)。

stdout用於print()的列印輸出或者input()函式的提示符。

stderr用於直譯器自己的提示資訊和錯誤資訊。

簡而言之,這三個屬性就是作業系統的標準輸入、輸出和錯誤流,它們返回的都是一個“檔案型別”物件,支援read()、write()和flush()等操作,就像用open()方法開啟的檔案物件那樣!

>>>import sys

>>>s=sys.stdin.read() # 使用ctrl+d結束輸入

i like python

end

>>>s

'i like python\nend\n\n\n\n'

>>>sys.stdout.write(s)

i like python

end

21

sys.stdout與print()

當我們print(obj)的時候,事實上是呼叫了sys.stdout.write(obj+'\n'),將內容列印到控制檯(預設是顯示器),然後追加一個換行符。以下兩行等價:

sys.stdout.write('hello'+'\n')

print('hello')

sys.stdin與input()

當我們用input('Please input something!')時,事實上是先輸出提示資訊,然後捕獲輸入。 以下兩組等價:

s=input('Please input something!')

print('Please input something!',) #逗號表示不換行s=sys.stdin.readline()[:-1] #-1可以拋棄輸入流中的'\n'換行符,自己琢磨一下為什麼。

8. 從控制檯重定向到檔案

預設情況下sys.stdout指向控制檯。如果把檔案物件賦值給sys.stdout,那麼print ()呼叫的就是檔案物件的write()方法。

f_handler=open('out.log','w')

sys.stdout=f_handler

print('hello')

# 你無法在螢幕上看到“hello”

# 因為它被寫到out.log檔案裡了

如果你還想同時在控制檯列印的話,最好先將原始的控制檯物件引用儲存下來,向檔案中列印之後再恢復 sys.stdout

__console__=sys.stdout # 儲存控制檯

# redirection start # 去幹點別的,比如寫到檔案裡

...

# redirection end # 幹完別的了,恢復原來的控制檯sys.stdout=__console__

9. 例項:帶百分比的進度條

利用sys.stdout的功能,可以實現一個簡易的進度條。

import sys

import time

def bar(num,total):

rate=num/total

rate_num=int(rate*100)

r='\r[%s%s]%d%%'%("="*num," "*(100-num),rate_num,)

sys.stdout.write(r)

sys.stdout.flush()

if__name__=='__main__':

for i in range(0,101):

time.sleep(0.1)

bar(i,100)