[ Python ] 模塊詳解

1. time 模塊

Functions:html

time() -- return current time in seconds since the Epoch as a float
	返回當前時間,以秒爲單位,從1870年開始就算,爲浮點數
clock() -- return CPU time since process start as a float
	以浮點數計算的秒數返回當前 CPU 的時間
sleep() -- delay for a number of seconds given as a float
	表示線程掛起的時間
gmtime() -- convert seconds since Epoch to UTC tuple
	將一個時間戳轉換爲 UTC 時間的 struct_time
localtime() -- convert seconds since Epoch to local time tuple
	格式化時間戳爲本地時間 struct_time
asctime() -- convert time tuple to string
	接收時間元組並返回一個可讀形式的時間
ctime() -- convert time in seconds to string
	把時間戳轉換爲 asctime() 的形式
mktime() -- convert local time tuple to seconds since Epoch
	接收 struct_time 對象做爲參數,返回用秒數來表示時間的浮點數
strftime() -- convert time tuple to string according to format specification
	函數接收以時間元組,並返回可讀字符串表示當地時間,格式由參數format決定
strptime() -- parse string to time tuple according to format specification
	函數根據指定的格式把一個時間字符串解析爲時間元組
tzset() -- change the local timezone
	根據環境變量TZ從新初始化時間相關設置

 

 

三種時間類型之間的轉換:python

time_stamp: 時間戳,表現形式: 1531433179.1281905 秒做爲單位的浮點數
struct_time:  結構化時間,表示形式:time.struct_time(tm_year=2018, tm_mon=7, tm_mday=13, tm_hour=6, tm_min=6, tm_sec=19, tm_wday=4, tm_yday=194, tm_isdst=0)
format_string: 格式化時間,表示形式:2018-07-13 06:11:02

 

時間轉換的詳細示意圖:web

 

除了上面時間轉換的相關方法,還有幾個須要注意的方法:算法

 

(1)time.asctime
    打印 '%a %b %d %H:%M:%S %Y' 可讀的時間格式shell

>>> print(time.asctime())
Fri Jul 13 06:31:32 2018

 

(2)time.ctime
       將時間戳轉換格式爲  '%a %b %d %H:%M:%S %Y' 可讀時間編程

>>> time.ctime(time.time())
'Fri Jul 13 06:33:56 2018'

 

(3)time.sleep
    進程掛起的時間,參數單位爲秒json

>>> time.sleep(2)

 

2. random 模塊

用於生成隨機數的模塊bash

 經常使用方法:網絡

(1)random.random
    隨機生成 0-1 之間的隨機數dom

>>> random.random()
0.34573032308214957

 

(2)random.randint
    傳入參數必須是整數,隨機選取參數範圍內的整數,首尾均可以取

>>> random.randint(0, 100)
30

 

(3)random.randrange
    傳入參數必須是整數,隨機選取參數範圍內的整數,首取尾不取
    只會在1,2之間隨機

>>> random.randrange(1, 3)
1

 

 (4)random.sample
    參數的第一個參數是可迭代對象,第二個參數是隨機獲取元素的個數,返回列表類型

>>> random.sample((1, '23', [4,5]), 2)
[[4, 5], 1]
>>> str1 = random.sample((1, '23', [4,5]), 2)
>>> type(str1)
<class 'list'>

 

(5)random.choice
    參數爲可迭代對象,隨機返回其中的一個元素

>>> random.choice(['a', 'b', 'c', 'xyz'])
'xyz'

 

 (6)random.uniform
    取任意整數範圍的浮點數

>>> random.uniform(1, 10)
8.77739630139806

 

 (7)random.shuffle
    打印列表或元組的順序,如洗牌

>>> item = [2,4,6,8,10]
>>> random.shuffle(item)
>>> item
[2, 8, 10, 6, 4]

 

實例:隨機驗證碼

import random

def v_code():
        '''隨機生成 4 位的驗證碼'''
        ran_code = ''
        for i in range(4):
                ran_int = random.randint(0, 9)  # 隨機獲取 0-9 之間任意一個整數
                ran_alf = chr(random.randint(65, 90))   # 隨機獲取 65-90之間一個隨機數,經過 chr 返回整數對應的 ASCII字符
                s = str(random.choice([ran_int, ran_alf]))  # 經過隨機 choice 選取其中一個
                ran_code += s   # 疊加字符串

 

 3. OS模塊

  OS模塊是與操做系統交互的一個接口

 

經常使用方法:

(1)os.getcwd()
    獲取當前工做目錄

>>> os.getcwd()
'C:\\Users\\hkey'

 

(2)os.chdir()
    改變當前腳本工做目錄,至關於 shell 'cd '命令

>>> os.chdir('..')
>>> os.getcwd()
'C:\\Users'

 

(3)os.curdir
    返回當前目錄

>>> os.curdir
'.'

 

(4)os.pardir
    返回當前目錄的父目錄字符串名

>>> os.pardir
'..'

 

 (5)os.makedirs('dirname1/dirname2')
    可生成多層遞歸目錄

>>> os.makedirs('abc/hkey')

 

 (6)os.removedirs('dirname1')
    若目錄爲空,則刪除,並遞歸到上一級目錄,如若也爲空,則刪除,以此類推

>>> os.removedirs('abc\hkey')
# 若是 hkey 目錄爲空則刪除,如 abc 目錄也空則刪除

 

 (7)os.mkdir('dirname')
    生成單級目錄;至關於shell中 mkdir dirname

>>> os.mkdir('hkey')

 

(8)os.rmdir('filename')
    刪除單級空目錄,若目錄不爲空則沒法刪除。

>>> os.rmdir('hkey')

 

(9)os.listdir('dirname')
    列出指定目錄下的全部文件和子目錄,包括隱藏文件,並以列表方式打印

>>> os.listdir(r'c:/python35')
['DLLs', 'Doc', 'include', 'Lib', 'libs', 'LICENSE.txt', 'NEWS.txt', 'python.exe', 'python3.dll', 'python35.dll', 'pythonw.exe', 'README.txt', 'Scripts', 'tcl', 'Tools', 'vcruntime140.dll']

 

(10)os.remove()
    刪除一個文件

>>> os.remove('abc.txt')

 

 (11)os.rename('oldname', 'newname')
    重命名文件、目錄

>>> os.rename('abc.txt', 'hkey.txt')

 

(12)os.stat('path/filename')
    獲取文件/目錄信息

>>> os.stat('hkey.txt')
os.stat_result(st_mode=33206, st_ino=562949953433293, st_dev=2283361558, st_nlink=1, st_uid=0, st_gid=0, st_size=0, st_atime=1531447766, st_mtime=1531447766, st_ctime=1531447766)

 

(13)os.sep
    輸出操做系統特定的路徑分隔符,win下爲 '\\', Linux下爲 '/'

>>> os.sep
'\\'

 

(14)os.linesep
    輸出當前平臺使用的行終止符,  win 下爲 '\r\n' Linux下爲'\n'

>>> os.linesep
'\r\n'

 

 (15)os.pathsep
    輸出環境變量 'path' 用於分割文件路徑的字符串 win下爲 ; Linux下爲 :

>>> os.pathsep
';'

 

 (16)os.name
    輸出字符串指定當前使用平臺, win --> 'nt' , Linux --> 'posix'

>>> os.name
'nt'

 

 (17)os.system('bash command')
    輸入shell命令,直接顯示不能保存變量

>>> os.system('dir C:\\')
  驅動器 C 中的卷沒有標籤。
  卷的序列號是 9483-7444

  C:\ 的目錄

2018/07/09  11:16    <DIR>          AMD
2018/07/09  11:23    <DIR>          DRIVERS
2018/07/09  11:15    <DIR>          Intel
2018/07/09  19:08    <DIR>          PerfLogs
2018/07/09  15:04    <DIR>          Program Files
2018/07/09  15:50    <DIR>          Program Files (x86)
2018/07/09  14:47    <DIR>          Python35
2018/07/09  14:12    <DIR>          Users
2018/07/11  11:19    <DIR>          Windows
                              0 個文件              0 字節
                              9 個目錄 68,863,184,896

 

(18)os.environ
    獲取系統環境變量

>>> os.environ
environ({'LOCALAPPDATA': 'C:\\Users\\hkey\\AppData\\Local'......})

 

 (19)os.path.abspath(path)
    返回 path 絕對路徑

>>> os.path.abspath('etc')
'C:\\Program Files\\Git\\etc'

 

 (20)os.path.split(path)
    將 path 分割爲 目錄 和 文件 2元素 元組

>>> os.path.split(r'C:\Program Files\Git\etc\host')
('C:\\Program Files\\Git\\etc', 'host')

 

(21)os.path.dirname(path)
    返回path的目錄。其實就是 os.path.split(path)的第一個元素

>>> os.path.dirname('C:\Program Files\Git\etc')
'C:\\Program Files\\Git'

 

(22)os.path.basename(path)
    返回 path 最後的文件名。如何 path 以 / 或 \\ 結尾,那麼就會返回空值。即 os.path.split(path) 的第二個元素

>>> os.path.basename('C:\Program Files\Git\etc\\')
''
>>> os.path.basename('C:\Program Files\Git\etc/')
''
>>> os.path.basename('C:\Program Files\Git\etc')
'etc'

 

(23)os.path.exists(path)
    若是 path 存在, 返回 True;若是 path 不存在, 返回 False

>>> os.path.exists('C:\Program Files\Git\etc')
True
>>> os.path.exists('C:\Program Files\Git\etcc')
False

 

(24)os.path.isabs(path)
    若是 path 是絕對路徑,返回 True

>>> os.path.isabs('etc')
False
>>> os.path.isabs('C:\Program Files\Git\etc')
True

 

(25)os.path.isfile(path)
    若是 path 是一個存在的文件,返回 True、不然返回 False

>>> os.path.isfile('C:\Program Files\Git\etc')
False
>>> os.path.isfile('C:\Program Files\Git\etc\hosts')
True

 

(26)os.path.isdir(path)
    若是 path 是一個存在的目錄,則返回 True,不然返回 False

>>> os.path.isdir('C:\Program Files\Git\etc')
True
>>> os.path.isdir('C:\Program Files\Git\etc\hosts')
False

 

 (27)os.path.join(path1[, path2[, …]])
    將多個路徑組合後返回,第一個絕對路徑以前的參數將被忽略

>>> os.path.join('\tmp\abc', 'C:\Program Files\Git\etc', r'conf\vhost')
'C:\\Program Files\\Git\\etc\\conf\\vhost'

 

(28)os.path.getatime(path)
    返回 path 所指向的文件或者目錄的最後存取時間

>>> os.path.getatime('C:\Program Files\Git\etc')
1531118531.7878392
>>> os.path.getatime('C:\Program Files\Git\etc\hosts')
1468669537.155324

 

(29)os.path.getmtime(path)
    返回 path 所指向的文件或者目錄的最後修改時間

>>> os.path.getmtime('C:\Program Files\Git\etc')
1531118531.7878392
>>> os.path.getmtime('C:\Program Files\Git\etc\hosts')
1468669537.155324

 

4. sys模塊

 (1) sys.argv
    命令行參數 List, 第一個元素是程序自己路徑

>python sys_test.py hello world
['sys_test.py', 'hello', 'world']

 

(2)sys.version
    獲取 python 解釋程序的版本信息

>>sys.version
'3.5.3 (v3.5.3:1880cb95a742, Jan 16 2017, 16:02:32) [MSC v.1900 64 bit (AMD64)]'

 

(3)sys.path
    返回模塊的搜索路徑,初始化時使用 python path 環境變量的值

sys.path
['D:\\Program Files (x86)\\JetBrains\\PyCharm 2017.3.3\\helpers\\pydev ......]

 

(4)sys.platform
    返回操做系統平臺名稱

>>> sys.platform
'win32'

 

實例:進度條程序

import sys, time

for i in range(100):
        sys.stdout.write('#')
        sys.stdout.flush()
        time.sleep(0.2)

 

5. json & pickle

 什麼是序列化:
    咱們把對象(變量)從內存中變成可存儲或傳輸的過程稱之爲序列化,在 python 中叫 picking
    序列化以後,就能夠把虛擬化後的內容寫入磁盤,或者經過網絡傳輸到別的機器上;
    反過來,把變量內容從序列化的對象從新讀到內存裏稱之爲反序列化,即 unpicking

 

json
    若是咱們要在不一樣的編程語言之間傳遞對象,就必須把對象序列化爲標準格式,好比 xml, 但更好的方法是虛擬化爲 JSON,由於JSON表示出來就是一個字符串,
    能夠被全部語言讀取,也能夠方便的存儲到磁盤或者經過網絡傳輸。JSON不只是標準格式,而且比xml更快,並且能夠直接在web頁面中讀取,很是方便。

 

    JSON表示的對象就是標準的Javascript語言對象,JSON和Python內置的數據類型對應以下:

 

 經常使用方法:
(1)json.dumps 序列化

(2)json.loads 反序列化

import json

# 序列化 -------------------------------
dic = {'name': 'hkey', 'age': 22}

j = json.dumps(dic)
print(j, type(j))

# 執行結果:
# {"age": 22, "name": "hkey"} <class 'str'>

# 反序列化 --------------------------------

i = json.loads(j)
print(i, type(i))

# 執行結果:
# {'age': 22, 'name': 'hkey'} <class 'dict'>

 

(3)json.dump
    序列化後寫入文件
    
(4)json.load
    讀取文件中json格式數據後,反序列化

import json
dic = {'name': 'hkey', 'age': 22}

# ----------- 序列化 -----------
with open('json.txt', 'w', encoding='utf-8') as f:
        # 序列化後,直接存儲 json.txt 文件
        json.dump(dic, f)

# ----------- 反序列化 -----------

with open('json.txt', 'r') as f:
        # 讀取文件中json類型數據並作反序列化
        data = json.load(f)
print(data, type(data))

# 執行結果:
# {'age':22,'name':'hkey'} <class'dict'>

 

 pickle 模塊

     做爲 python 特有的類型和 python 的數據類型間進行轉換
    
    (1)pickle.dumps
        以字節對象形式返回封裝的對象,不須要寫入文件中;
    (2)pickle.loads
        從字節對象中讀取被封裝的對象,並返回;

import pickle
dic = {'name': 'hkey', 'age': 22}

#------------序列化------------
j = pickle.dumps(dic)
print(j)

# 執行結果:
# b'\x80\x03}q\x00(X\x03\x00\x00\x00ageq\x01K\x16X\x04\x00\x00\x00nameq\x02X\x04\x00\x00\x00hkeyq\x03u.'

#------------反序列化------------
data = pickle.loads(j)
print(data, type(data))

# 執行結果:
# {'name': 'hkey', 'age': 22} <class 'dict'>

 

(3)pickle.dump
    必填參數 file 表示 obj 要寫入的文件對象,file必須以二進制可寫模式打開,即 'wb'
(4)pickle.load
        必填參數 file 必須以二進制可讀模式打開,即 'rb', 其餘均可選參數

import pickle
dic = {'name': 'hkey', 'age': 22}
#------------序列化------------
with open('pickel.txt', 'wb') as f:
        # 序列化後,以字節格式存入 pickel.txt 文件
        pickle.dump(dic, f)
#------------反序列化------------
with open('pickel.txt', 'rb') as f:
        # 讀取文件中字節,並作反序列化
        data = pickle.load(f)

print(data)

# 執行結果:
#{'name':'hkey','age':22}

 

6. shelve 模塊

 shelve 是一組簡單的數據存儲方案,他只有一個函數就是 open(), 這個函數接收一個參數就是文件名,而且文件名必須是 .dat類型的。
而後返回一個對象,能夠把這個對象看成一個字典,當你存儲完畢的時候,就調用 close 函數來關閉

寫入:

import shelve

f = shelve.open('shelve.txt')
f['key1'] = {'name': 'xiaofei', 'age': 20}
f.close()

 執行完畢會生成以下三個文件:

 

讀取:

import shelve

with shelve.open('shelve.txt') as f:
        name = f['key1']['name']
print(name)

# 執行結果:
# xiaofei

 

7. configparser模塊

 該模塊的做用就是使用模塊中的 ConfigParser(),建立一個對象使用對象的方法對指定的配置文件作 增刪改查 操做

(1 ) 寫入

import configparser

config = configparser.ConfigParser()
config['DEFAULT'] = {'ServerAliveInterval': '45',
                                            'Compression': 'yes',
                                          'CompressionLevel': '9'}

config['bitbucket.org'] = {}
config['bitbucket.org']['User'] = 'hg'
config['topsecret.server.com'] = {}
topsecret = config['topsecret.server.com']
topsecret['Host Port'] = '50022'
topsecret['ForwardX11'] = 'no'
config['DEFAULT']['ForwardX11'] = 'yes'

with open('exmaple.ini', 'w') as configfile:
        config.write(configfile)

 

 (2)讀取

import configparser
config = configparser.ConfigParser()
config.read('example.ini')
print(config.sections())    # ['bitbucket.org', 'topsecret.server.com']
print('bytebong.com' in config)     # False
print(config['bitbucket.org']['User'])  # hg
print(config['DEFAULT']['Compression'])     # yes
print(config['topsecret.server.com']['ForwardX11'])     # no

for key in config['bitbucket.org']:
        print(key)

# user
# serveraliveinterval
# compression
# compressionlevel
# forwardx11

 

(3)增長

import configparser

config = configparser.ConfigParser()
config.read('example.ini')
config.add_section('xiaofei')
config['xiaofei'] = {
        'name': 'xiaofei',
        'age': 20
}

config.write(open('example.ini', 'w'))

 

(4)修改、刪除

import configparser

config = configparser.ConfigParser()
config.read('example.ini')
config.remove_section('bitbucket.org')      # 刪除 [bitbucket.org] 下全部內容
config.remove_option('xiaofei', 'name')     # 刪除 [xiaofei] 下 name項
config.set('xiaofei', 'age', '18')          # 修改 [xiaofei] age = 18 必須爲str類型
config.write(open('i.ini', 'w'))            # 保存修改內容

 

8. hashlib模塊

  用於加密相關操做。主要用 md5 進行加密

import hashlib

m = hashlib.md5()
m.update('hello'.encode('utf-8'))
print(m.hexdigest())    # 5d41402abc4b2a76b9719d911017c592
m.update('world'.encode('utf-8'))
print(m.hexdigest())    # fc5e038d38a57032085441e7fe7010b0

m2 = hashlib.md5()
m2.update('helloworld'.encode('utf-8'))
print(m2.hexdigest())   # fc5e038d38a57032085441e7fe7010b0

 以上加密存在缺陷,能夠經過撞庫來反解,因此有必要對加密算法中添加自定義 key 再來作加密

import hashlib

hash = hashlib.sha256('888'.encode('utf-8'))
hash.update('aliyun'.encode('utf-8'))
print(hash.hexdigest()) #da7ecd435e6e0930532c115e7fe48c38d0405aa79586b0275717a0ab0a85acd1

 

9. logging模塊

  用於記錄日誌的模塊

import logging

logging.debug('debug message')
logging.info('info message')
logging.warning('warning message')
logging.error('error message')
logging.critical('critical message')

# 執行結果:
# WARNING:root:warning message
# ERROR:root:error message
# CRITICAL:root:critical message

 

默認狀況下 python 的 logging 模塊將日誌打印到了標準輸出中,且只顯示了大於 WARING級別的日誌,這說明默認的日誌級別設置爲 WARING
日誌級別等級排序(CRITICAL > ERROR > WARING > INFO > DEBUG > NOTSET)默認的日誌格式爲日誌級別: Logger名稱:用戶輸出消息

(1)靈活配置日誌級別,日誌格式,輸出位置

import logging

logging.basicConfig(
        level=logging.DEBUG,
        format='%(asctime)s %(filename)s %(message)s',
        # filename='logg.txt',
        # filemode='w'
)

logging.debug('debug message')
logging.info('info message')
logging.warning('warning message')
logging.error('error message')
logging.critical('critical message')

# 執行結果:
# 2018-07-14 08:51:56,441 logging_test.py debug message
# 2018-07-14 08:51:56,441 logging_test.py info message
# 2018-07-14 08:51:56,441 logging_test.py warning message
# 2018-07-14 08:51:56,441 logging_test.py error message
# 2018-07-14 08:51:56,441 logging_test.py critical message

 

能夠在 logging.basicConfig() 函數中經過具體的參數來更改 logging 模塊默認行爲,可用參數以下:

filename:用指定的文件名建立FiledHandler(後邊會具體講解handler的概念),這樣日誌會被存儲在指定的文件中。
filemode:文件打開方式,在指定了filename時使用這個參數,默認值爲「a」還可指定爲「w」。
format:指定handler使用的日誌顯示格式。 
datefmt:指定日期時間格式。 
level:設置rootlogger(後邊會講解具體概念)的日誌級別 
stream:用指定的stream建立StreamHandler。能夠指定輸出到sys.stderr,sys.stdout或者文件(f=open('test.log','w')),默認爲sys.stderr。若同時列出了filename和stream兩個參數,則stream參數會被忽略。

format參數中可能用到的格式化串:
%(name)s Logger的名字
%(levelno)s 數字形式的日誌級別
%(levelname)s 文本形式的日誌級別
%(pathname)s 調用日誌輸出函數的模塊的完整路徑名,可能沒有
%(filename)s 調用日誌輸出函數的模塊的文件名
%(module)s 調用日誌輸出函數的模塊名
%(funcName)s 調用日誌輸出函數的函數名
%(lineno)d 調用日誌輸出函數的語句所在的代碼行
%(created)f 當前時間,用UNIX標準的表示時間的浮 點數表示
%(relativeCreated)d 輸出日誌信息時的,自Logger建立以 來的毫秒數
%(asctime)s 字符串形式的當前時間。默認格式是 「2003-07-08 16:49:45,896」。逗號後面的是毫秒
%(thread)d 線程ID。可能沒有
%(threadName)s 線程名。可能沒有
%(process)d 進程ID。可能沒有
%(message)s用戶輸出的消息

 

 (2)logger 對象

import logging

logger = logging.getLogger()

fh = logging.FileHandler('test.log')

ch = logging.StreamHandler()

formatter = logging.Formatter('%(asctime)s - %(levelname)s - %(message)s')

fh.setFormatter(formatter)
ch.setFormatter(formatter)

logger.addHandler(fh)
logger.addHandler(ch)

logger.debug('logger debug message')
logger.info('logger info message')
logger.warning('logger warning message')
logger.error('logger error message')
logger.critical('logger critical message')

# 執行結果:
# 2018-07-14 09:08:54,983 - WARNING - logger warning message
# 2018-07-14 09:08:54,983 - ERROR - logger error message
# 2018-07-14 09:08:54,983 - CRITICAL - logger critical message

 

代碼結構:

 

 logging庫提供了多個組件:
    Logger: 對象提供應用程序可直接使用的接口
    Handler:發送日誌到適當的目的地
    Filter: 提供了過濾日誌信息的方法
    Formatter: 指定日誌顯示的格式

logger = logging.getLogger() 返回一個默認的 Logger 也是 root Logger,並應用默認的日誌級別、Handler 和 Formatter 設置
能夠經過指定最低的日誌級別來顯示日誌信息:
    Logger.DEBUG
    Logger.INFO
    Logger.WARING
    Logger.ERROR
    Logger.CRITICAL

# 執行結果:
# 2018-07-14 09:08:54,983 - WARNING - logger warning message
# 2018-07-14 09:08:54,983 - ERROR - logger error message
# 2018-07-14 09:08:54,983 - CRITICAL - logger critical message

 從這個輸出能夠看出 logger = logging.getLogger() 返回 Logger名爲 root. 這裏沒有用 logger.setLevel(logging.Debug) 顯示的爲 logger 設置日誌級別,因此使用默認的日誌級別
WARING,故結果只輸出了大於等於 WARING 級別的信息

 

若是建立兩個 logger 對象

import logging

logger = logging.getLogger()
# 建立一個handler,用於寫入日誌文件
fh = logging.FileHandler('test.log')

# 再建立一個handler,用於輸出到控制檯
ch = logging.StreamHandler()

formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s')

fh.setFormatter(formatter)
ch.setFormatter(formatter)

logger.addHandler(fh) #logger對象能夠添加多個fh和ch對象
logger.addHandler(ch)

logger.debug('logger debug message')
logger.info('logger info message')
logger.warning('logger warning message')
logger.error('logger error message')
logger.critical('logger critical message')
    
##################################################
logger1 = logging.getLogger('mylogger')
logger1.setLevel(logging.DEBUG)

logger2 = logging.getLogger('mylogger')
logger2.setLevel(logging.INFO)

logger1.addHandler(fh)
logger1.addHandler(ch)

logger2.addHandler(fh)
logger2.addHandler(ch)

logger1.debug('logger1 debug message')
logger1.info('logger1 info message')
logger1.warning('logger1 warning message')
logger1.error('logger1 error message')
logger1.critical('logger1 critical message')

logger2.debug('logger2 debug message')
logger2.info('logger2 info message')
logger2.warning('logger2 warning message')
logger2.error('logger2 error message')
logger2.critical('logger2 critical message')
    
# 執行結果:
# 2018-07-14 09:30:39,054 - root - WARNING - logger warning message
# 2018-07-14 09:30:39,054 - root - ERROR - logger error message
# 2018-07-14 09:30:39,054 - root - CRITICAL - logger critical message
# 2018-07-14 09:30:39,054 - mylogger - INFO - logger1 info message
# 2018-07-14 09:30:39,054 - mylogger - INFO - logger1 info message
# 2018-07-14 09:30:39,055 - mylogger - WARNING - logger1 warning message
# 2018-07-14 09:30:39,055 - mylogger - WARNING - logger1 warning message
# 2018-07-14 09:30:39,055 - mylogger - ERROR - logger1 error message
# 2018-07-14 09:30:39,055 - mylogger - ERROR - logger1 error message
# 2018-07-14 09:30:39,055 - mylogger - CRITICAL - logger1 critical message
# 2018-07-14 09:30:39,055 - mylogger - CRITICAL - logger1 critical message
# 2018-07-14 09:30:39,055 - mylogger - INFO - logger2 info message
# 2018-07-14 09:30:39,055 - mylogger - INFO - logger2 info message
# 2018-07-14 09:30:39,055 - mylogger - WARNING - logger2 warning message
# 2018-07-14 09:30:39,055 - mylogger - WARNING - logger2 warning message
# 2018-07-14 09:30:39,055 - mylogger - ERROR - logger2 error message
# 2018-07-14 09:30:39,055 - mylogger - ERROR - logger2 error message
# 2018-07-14 09:30:39,055 - mylogger - CRITICAL - logger2 critical message
logger

 

 

這裏有兩個問題:
    明明經過 logger1.setLevel(logging.DEBUG) 將 logger1的日誌級別設置爲 DEBUG,爲什麼顯示的時候沒有顯示出 DEBUG 級別的日誌信息?
    
    由於 logger1 和 logger2 對應的是同一個 Logger 實例,只要 logging.getLogger( name ) 中名稱參數 name 相同則返回的 Logger實例就是同一個,且僅有一個
    在 logger2實例中經過 logger2.setLevel(logging.INFO)設置mylogger的日誌級別爲 logging.INFO,因此最後 logger1 的輸出聽從了後來設置的日誌級別。
    
    爲何 logger一、logger2 對應的每一個輸出分別顯示兩次?
    這是由於咱們經過 logger=logging.getLogger() 顯示的建立了 root Logger , 而 logger1 = logging.getLogger('mylogger') 建立了 root logger的孩子,logger2一樣
    那麼孩子就會將消息分發給他的 handler 進行處理也會傳遞給全部的祖先 Logger 處理
    
註釋掉 root 的 handler 處理:

import logging

logger = logging.getLogger()
# 建立一個handler,用於寫入日誌文件
fh = logging.FileHandler('test.log')

# 再建立一個handler,用於輸出到控制檯
ch = logging.StreamHandler()

formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s')

fh.setFormatter(formatter)
ch.setFormatter(formatter)

# logger.addHandler(fh) #logger對象能夠添加多個fh和ch對象
# logger.addHandler(ch)

logger.debug('logger debug message')
logger.info('logger info message')
logger.warning('logger warning message')
logger.error('logger error message')
logger.critical('logger critical message')

##################################################
logger1 = logging.getLogger('mylogger')
logger1.setLevel(logging.DEBUG)

logger2 = logging.getLogger('mylogger')
logger2.setLevel(logging.INFO)

logger1.addHandler(fh)
logger1.addHandler(ch)

logger2.addHandler(fh)
logger2.addHandler(ch)
        
logger1.debug('logger1 debug message')
logger1.info('logger1 info message')
logger1.warning('logger1 warning message')
logger1.error('logger1 error message')
logger1.critical('logger1 critical message')

logger2.debug('logger2 debug message')
logger2.info('logger2 info message')
logger2.warning('logger2 warning message')
logger2.error('logger2 error message')
logger2.critical('logger2 critical message')

# 執行結果:
# logger warning message
# logger error message
# logger critical message
# 2018-07-14 09:41:03,604 - mylogger - INFO - logger1 info message
# 2018-07-14 09:41:03,604 - mylogger - WARNING - logger1 warning message
# 2018-07-14 09:41:03,604 - mylogger - ERROR - logger1 error message
# 2018-07-14 09:41:03,604 - mylogger - CRITICAL - logger1 critical message
# 2018-07-14 09:41:03,604 - mylogger - INFO - logger2 info message
# 2018-07-14 09:41:03,604 - mylogger - WARNING - logger2 warning message
# 2018-07-14 09:41:03,604 - mylogger - ERROR - logger2 error message
logger

 

 

10.  re模塊

  請參考:

    -------------re模塊連接-------------

相關文章
相關標籤/搜索