random模塊用于獲取隨機(jī)數(shù),一下random模塊中常用的函數(shù):
網(wǎng)站建設(shè)哪家好,找創(chuàng)新互聯(lián)公司!專(zhuān)注于網(wǎng)頁(yè)設(shè)計(jì)、網(wǎng)站建設(shè)、微信開(kāi)發(fā)、微信小程序、集團(tuán)企業(yè)網(wǎng)站建設(shè)等服務(wù)項(xiàng)目。為回饋新老客戶(hù)創(chuàng)新互聯(lián)還提供了懷化免費(fèi)建站歡迎大家使用!
# 返回 (0,1) ,float類(lèi)型
random.random()
# 返回 [1,3],int 類(lèi)型
random.randint(1, 3)
# 返回 [1,3),int 類(lèi)型
random.randrange(1, 3)
# 隨機(jī)獲取列表中的一個(gè)元素
random.choice([3,4,5,2,1, 'kitty'])
# 隨機(jī)獲取列表中的2個(gè)元素,以列表的形式返回
random.sample([3,4,5,2,1, 'kitty'], 2)
# 返回[1,3],float類(lèi)型
random.uniform(1,3)
# 隨機(jī)打亂 類(lèi)表lst 中的元素順序
lst = [111,222,333,444]
random.shuffle(lst)
示例(隨機(jī)獲取驗(yàn)證碼),5位驗(yàn)證碼,包含整數(shù),大小寫(xiě)字母~
def valdate_code():
res = ''
for i in range(5):
num = random.randint(0, 9)
alpha_lower = chr(random.randint(97, 122)) # 小寫(xiě)字母
alpha_upper = chr(random.randint(65, 90)) # 大寫(xiě)字母
s = random.choice([str(num), alpha_lower, alpha_upper])
res += s
return res
調(diào)用結(jié)果:
8Rj0x
306GX
...
hashlib模塊提供了常見(jiàn)的摘要算法,如MD5,SHA1等。
摘要算法是指 通過(guò)一個(gè)函數(shù),將任意長(zhǎng)度的數(shù)據(jù)轉(zhuǎn)換為一個(gè)固定長(zhǎng)度的字符串,通常用16進(jìn)制的字符串表示~
import hashlib
md5_obj = hashlib.md5()
md5_obj.update(b"hello world")
print(md5_obj.hexdigest()) # 5eb63bbbe01eeed093cb22bb8f5acdc3
# 現(xiàn)在對(duì)hello world改變一個(gè)字母
md5_obj.update(b"hello World")
print(md5_obj.hexdigest()) # 4245dd40eaf3111caa3c8f9e3ceeed3c
數(shù)據(jù)由 'hello world' 改成 'hello World',獲取的摘要信息完全不相同。所以摘要算法一般用于提取數(shù)據(jù)的特征碼。
摘要函數(shù)是一個(gè)單向函數(shù),通過(guò)數(shù)據(jù)計(jì)算出其特征碼很容易,但是要根據(jù)特征碼反推出數(shù)據(jù)卻很困難。通過(guò)從數(shù)據(jù)中提取出的特征碼可以判斷數(shù)據(jù)是否被篡改過(guò)~
若現(xiàn)在要獲取一個(gè)大文件的特征碼,可以每讀取一行,對(duì)這一行的數(shù)據(jù)進(jìn)行一次update(),到最后再進(jìn)行特征碼的計(jì)算,示例如下:
import hashlib
md5_obj = hashlib.md5()
with open(file='/Users/luyi/tmp/passwd', mode='r', encoding='utf-8') as f:
for line in f:
md5_obj.update(line.encode('utf-8'))
print(md5_obj.hexdigest())
Tip:在python3中,傳遞給update的參數(shù)必須是 bytes 類(lèi)型。python3中字符串默認(rèn)使用 unicode 形式保存在內(nèi)存中,需要將 unicode 形式的字符串 encode 為 bytes 類(lèi)型再進(jìn)行操作~
?
上述示例中使用的摘要算法都是md5,md5是常見(jiàn)的摘要算法,生成速度快,生成的結(jié)果是一個(gè)固定的128 bit字節(jié),通常用一個(gè)32位的16進(jìn)制字符串表示。除了md5還有一種摘要算法sha1,調(diào)用sha1的方式與調(diào)用md5類(lèi)似,sha1的結(jié)果是160 bit字節(jié),通常用一個(gè)40位的16進(jìn)制字符串表示。
import hashlib
sha1_obj = hashlib.sha1()
sha1_obj.update(b'hello world')
sha1_obj.update(b'hello kitty')
print(sha1_obj.hexdigest()) # 563258876190465d493543b96306a92164ac7e62
除了md5,sha1算法,還有 sha256 和 sha512,這兩個(gè)摘要算法獲取的摘要長(zhǎng)度更長(zhǎng),更安全,但是計(jì)算的速度會(huì)更慢~
?
獲取數(shù)據(jù)的特征碼,數(shù)據(jù)的長(zhǎng)度是任意的,但是獲取的特征碼(摘要信息)的長(zhǎng)度是固定的,那就有可能出現(xiàn)這種情況,兩個(gè)不一樣的數(shù)據(jù),提取的特征碼是一致的,這種情況稱(chēng)為碰撞,只是發(fā)生的概率不大~
摘要算法還通常用于密碼的保存,密碼先進(jìn)行單向加密后,然后再保存到數(shù)據(jù)庫(kù)中。當(dāng)需要驗(yàn)證密碼時(shí),將用戶(hù)輸入的密碼也進(jìn)行單向加密,然后和數(shù)據(jù)庫(kù)中存儲(chǔ)的進(jìn)行比對(duì)~
?
但是這樣就又有一個(gè)問(wèn)題,若用戶(hù)設(shè)置的密碼過(guò)于簡(jiǎn)單,例如很多人會(huì)使用 '123456','admin','password'這樣的密碼,若公司存放用戶(hù)信息的表丟失,嘿客可以事先計(jì)算出這些簡(jiǎn)單密碼的md5值,然后與表中加密后的密碼進(jìn)行比對(duì),這樣部分用戶(hù)的密碼就會(huì)被嘿客獲取。
e10adc3949ba59abbe56e057f20f883e 123456
21232f297a57a5a743894a0e4a801fc3 admin
5f4dcc3b5aa765d61d8327deb882cf99 password
?
解決的方法就是,對(duì)原始的密碼“加鹽操作”。即對(duì)原始的密碼再加上其特有的字符串,例如將用戶(hù)的密碼再加上其用戶(hù)名,然后再進(jìn)行單項(xiàng)加密操作~,這樣即使用戶(hù)使用的密碼相同,加上用戶(hù)名后獲取的摘要信息也不會(huì)相同~
import hashlib
md5_obj = hashlib.md5(b'kitty') # 在這里進(jìn)行加鹽
md5_obj.update(b"123456")
print(md5_obj.hexdigest())
os 模塊是與操作系統(tǒng)交互的一個(gè)接口
常用方法如下:
os.getcwd() # 獲取當(dāng)前工作目錄,即當(dāng)前python腳本工作的目錄路徑
os.chdir("dirname") # 改變當(dāng)前腳本工作目錄;相當(dāng)于shell下cd
os.curdir # 返回當(dāng)前目錄: ('.')
os.pardir # 獲取當(dāng)前目錄的父目錄字符串名:('..’)
os.makedirs('dirname1/dirname2') # 可生成多層遞歸目錄
os.removedirs('dirname1') # 若目錄為空,則刪除,并遞歸到上一級(jí)目錄,如若也為空,則刪除,依此類(lèi)推
os.mkdir('dirname') # 生成單級(jí)目錄;相當(dāng)于shell中mkdir dirname
os.rmdir('dirname') # 刪除單級(jí)空目錄,若目錄不為空則無(wú)法刪除,報(bào)錯(cuò);相當(dāng)于shell中rmdir dirname
os.listdir('dirname') #列出指定目錄下的所有文件和子目錄,包括隱藏文件,并以列表方式打印
os.remove() # 刪除一個(gè)文件
os.rename("oldname","newname") # 重命名文件/目錄,注意若文件未打開(kāi)狀態(tài),則無(wú)法rename
os.stat('path/filename') # 獲取文件/目錄信息
os.sep # 輸出操作系統(tǒng)特定的路徑分隔符,win下為"\\",Linux下為"/"
os.linesep # 輸出當(dāng)前平臺(tái)使用的行終止符,win下為"\t\n",Linux下為"\n"
os.pathsep # 輸出用于分割文件路徑的字符串 win下為;,Linux下為:
os.name # 輸出字符串指示當(dāng)前使用平臺(tái)。win->'nt'; Linux->'posix'
os.system("bash command") # 運(yùn)行shell命令,直接顯示
os.environ # 獲取系統(tǒng)環(huán)境變量
os.path.abspath(path) # 返回path規(guī)范化的絕對(duì)路徑
os.path.split(path) # 將path分割成目錄和文件名二元組返回
os.path.dirname(path) # 返回path的目錄。其實(shí)就是os.path.split(path)的第一個(gè)元素
os.path.basename(path) # 返回path最后的文件名。如何path以/或\結(jié)尾,那么就會(huì)返回空值。即os.path.split(path)的第二個(gè)元素
os.path.exists(path) # 如果path存在,返回True;如果path不存在,返回False
os.path.isabs(path) # 如果path是絕對(duì)路徑,返回True
os.path.isfile(path) # 如果path是一個(gè)存在的文件,返回True。否則返回False
os.path.isdir(path) # 如果path是一個(gè)存在的目錄,則返回True。否則返回False
os.path.join(path2[, path3[, ...]]) # 將多個(gè)路徑組合后返回,第一個(gè)絕對(duì)路徑之前的參數(shù)將被忽略
os.path.getatime(path) # 返回path所指向的文件或者目錄的最后訪(fǎng)問(wèn)時(shí)間
os.path.getmtime(path) # 返回path所指向的文件或者目錄的最后修改時(shí)間
os.path.getsize(path) # 返回path的大小
經(jīng)常會(huì)使用的就如下幾個(gè):
os.remove()
os.path.abspath(path)
os.path.abspath(__file__) # 獲取當(dāng)前執(zhí)行腳本的路徑
os.path.dirname(path)
os.path.basename(path)
os.path.exists(path)
os.path.isfile(path)
os.path.isdir(path)
os.path.join(path2[, path3[, ...]])
os.path.join('/','etc', 'passwd') # /etc/passwd
os.path.getsize(path)
os.path.getsize('/etc/passwd') # 6804,單位字節(jié)
os.access() 用來(lái)檢測(cè)指定路徑的訪(fǎng)問(wèn)權(quán)限~
語(yǔ)法:os.access(path, mode);
- path:用來(lái)檢測(cè)的路徑,可以是目錄,也可以是文件~
- mode:
- os.F_OK: 測(cè)試path是否存在。
- os.R_OK: 測(cè)試path是否可讀。
- os.W_OK: 測(cè)試path是否可寫(xiě)。
- os.X_OK: 測(cè)試path是否可執(zhí)行。
示例:
import os
# 判斷目錄是否存在
res = os.access("/tmp", os.F_OK)
print "F_OK - %s"% res
# 判斷文件是否存在
res = os.access("/tmp/abc", os.F_OK)
print "F_OK - %s"% res
# 判斷目錄是否可讀
res = os.access("/tmp", os.R_OK)
print "R_OK - %s"% res
# 判斷文件是否可讀
res = os.access("/tmp/abc", os.R_OK)
print "R_OK - %s"% res
# 判斷文件是否可寫(xiě)
res = os.access("/tmp/abc", os.W_OK)
print "W_OK - %s"% res
# 判斷文件是否可執(zhí)行
res = os.access("/tmp/abc", os.X_OK)
print "X_OK - %s"% res
# 執(zhí)行結(jié)果:
F_OK - True
F_OK - True
R_OK - True
R_OK - True
W_OK - True
X_OK - False
os 模塊是和操作系統(tǒng)交互的模塊,這里的sys是和python解釋器交互的模塊
列出常用方法即可
sys常用的方法如下:
sys.argv # 命令行參數(shù),以L(fǎng)ist形式返回,第一個(gè)元素是程序本身路徑
sys.exit(n) # 退出程序,正常退出是exit(0),參數(shù)為返回碼
sys.version # 獲取Python解釋程序的版本信息
sys.maxint # 最大的Int值
sys.path # 返回模塊的搜索路徑,初始化時(shí)使用PYTHONPATH環(huán)境變量的值
sys.platform # 返回操作系統(tǒng)平臺(tái)名稱(chēng)
sys.getdefaultencoding() # 獲取系統(tǒng)當(dāng)前默認(rèn)編碼,python2默認(rèn)為ascii,python3默認(rèn)為utf-8。
sys.setdefaultencoding() # python2中設(shè)置系統(tǒng)默認(rèn)編碼,執(zhí)行dir(sys)時(shí)不會(huì)看到這個(gè)方法,在解釋器中執(zhí)行不通過(guò),需要先執(zhí)行reload(sys),再進(jìn)行設(shè)置。python3中沒(méi)有此方法,也不能reload(sys)~
sys.getfilesystemencoding() # 獲取文件系統(tǒng)使用編碼方式,Windows下返回'mbcs',mac下返回'utf-8'.
sys.stdin,sys.stdout,sys.stderr # stdin , stdout , 以及 stderr 變量包含與標(biāo)準(zhǔn) I/O 流對(duì)應(yīng)的流對(duì)象. 如果需要更好地控制輸出,而 print 不能滿(mǎn)足要求, 可以使用stdin , stdout , stderr 替換。這時(shí)候可以重定向輸出或者輸入到其它設(shè)備( device ),或者以非標(biāo)準(zhǔn)的方式處理它們~
通過(guò)命令行運(yùn)行Python程序時(shí),命令行的執(zhí)行文件 及參數(shù)會(huì)以列表的形式存放在 sys.argv 變量中~
sys_test.py文件內(nèi)容如下:
import sys
print(sys.argv)
命令行執(zhí)行:
? ~ python ~/tmp/sys_test.py 1 2 3 4 5 6
['/Users/luyi/tmp/sys_test.py', '1', '2', '3', '4', '5', '6']
程序執(zhí)行完成后,python解釋器自動(dòng)退出,若由于某些原因需要在中途退出,可以使用 sys.exit(n) ,參數(shù)n 可指定退出時(shí)的狀態(tài)碼,一般n=0表示正常退出,其他數(shù)值(1-127)為非正常退出~
示例:
# 執(zhí)行如下內(nèi)容的py文件
import sys
sys.exit(2)
? tmp python sys_test.py
? tmp echo $?
2 # 狀態(tài)返回碼為 2
程序的中途退出也可以使用 SystemExit 進(jìn)行捕獲,在 except 中完成退出之前必要的事項(xiàng)
print('start...')
try:
sys.exit(1)
except SystemExit:
print('end...')
sys.exit(0)
print('contimue')
# 輸出結(jié)果:
start...
end...
sys.path 是一個(gè)列表,里面存放的是模塊的搜索路徑。若需要使用的模塊不在這些路徑中,可以直接將路徑添加到這個(gè)變量中,程序中的 import 就能正確導(dǎo)入該模塊~
>>> import sys
>>> sys.path
['', '/Library/Frameworks/Python.framework/Versions/3.6/lib/python36.zip', '/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6', '/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/lib-dynload', '/Library/Frameworks/Python.framework/Versions/3.6/lib/python3.6/site-packages']
當(dāng) python程序中涉及到 unicode類(lèi)型 和 編碼的字符串 相互轉(zhuǎn)換時(shí) (python2 中為 str 類(lèi)型 與 unicode類(lèi)型相互轉(zhuǎn)換,python3中為 str類(lèi)型 和 bytes類(lèi)型 之間的相互轉(zhuǎn)換,這一塊的詳細(xì)內(nèi)容可參見(jiàn) https://blog.51cto.com/ljbaby/2164480 ) 就會(huì)使用getdefaultencoding輸出的編碼進(jìn)行轉(zhuǎn)換~
python2中默認(rèn)編碼為 ascii,python3中默認(rèn)編碼為 utf-8 ~
# python2
import sys
print sys.getdefaultencoding()
輸出結(jié)果:
ascii
# python3
import sys
print(sys.getdefaultencoding())
輸出結(jié)果:
utf-8
python3中,當(dāng)str類(lèi)型(python3中字符串一律使用unicode存放)和 bytes類(lèi)型 合并時(shí),會(huì)直接報(bào)錯(cuò):
x = '你好,' # str類(lèi)型
y = '貝貝'.encode('utf-8') # bytes類(lèi)型
print(x + y)
報(bào)錯(cuò)信息:
TypeError: must be str, not bytes
但是在python2中,這個(gè)過(guò)程可以進(jìn)行,Python解釋器會(huì)自動(dòng)把 str 轉(zhuǎn)換成 unicode 再進(jìn)行運(yùn)算,運(yùn)算結(jié)果也都是 unicode類(lèi)型,在Python解釋器自動(dòng)將 str 轉(zhuǎn)成 unicode時(shí),由于沒(méi)有具體指定使用哪種編碼進(jìn)行轉(zhuǎn)碼,所以python解釋器就會(huì)默認(rèn)使用 getdefaultencoding中的編碼,python2中默認(rèn)編碼是ascii,于是就出現(xiàn)如下錯(cuò)誤:
x = u'你好,'
y = '貝貝'
print x + y
錯(cuò)誤信息:
UnicodeDecodeError: 'ascii' codec can't decode byte 0xe8 in position 0: ordinal not in range(128)
設(shè)置一下默認(rèn)編碼,就可以正常輸出:
# -*- coding: utf-8 -*-
import sys
reload(sys)
sys.setdefaultencoding('utf-8')
x = u'你好,'
y = '貝貝'
print x + y
輸出結(jié)果:
你好,貝貝
Tip:這個(gè)經(jīng)常在 python2 中使用,python3中沒(méi)有這樣的操作,sys不能 reload,也沒(méi)有 setdefaultencoding 方法~,python3中默認(rèn)編碼為utf-8,也不需要修改~
.................^_^