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)