1.引入線(xiàn)程
?????????所有的進(jìn)程默認(rèn)都有一個(gè)線(xiàn)程(一般這個(gè)線(xiàn)程為主線(xiàn)程), 有時(shí)需要在程序執(zhí)行的過(guò)程中要做多件事情,但是又不會(huì)影響主程序的運(yùn)行.(比如:在播放器聽(tīng)歌的時(shí)候,下載別的歌曲),這時(shí)候就要添加線(xiàn)程.
如果想要在進(jìn)程中添加其他的線(xiàn)程,就創(chuàng)建線(xiàn)程對(duì)象
????????python內(nèi)置的threading模塊,可以支持多線(xiàn)程
2.使用多線(xiàn)程
-
方法1:使用thread.Thread()方法,傳遞參數(shù)創(chuàng)建線(xiàn)程對(duì)象
t1 = threading.Thread(target=download, args=['愛(ài)情公寓'])
"""
target:需要在子線(xiàn)程中執(zhí)行的函數(shù)
args:調(diào)用函數(shù)的實(shí)參列表(參數(shù)類(lèi)型是列表)
返回值:先成對(duì)象
"""
實(shí)現(xiàn)方式:
import threading
import time
def download(file):
print('開(kāi)始下載', file)
time.sleep(5)
print(file, '下載成功')
if __name__ == '__main__':
print('abc')
# 1.創(chuàng)建線(xiàn)程對(duì)象
t1 = threading.Thread(target=download, args=['愛(ài)情公寓'])
# 2.在子線(xiàn)程中執(zhí)行任務(wù)
t1.start()
t2 = threading.Thread(target=download, args=['狄仁杰'])
t2.start()
print('======')
#輸出
F:\Pythonworkspace\day18\多線(xiàn)程\venv\Scripts\python.exe F:/Pythonworkspace/day18/多線(xiàn)程/01-多線(xiàn)程技術(shù).py
abc
開(kāi)始下載 愛(ài)情公寓
開(kāi)始下載 狄仁杰
======
狄仁杰 下載成功
愛(ài)情公寓 下載成功
此方法是自己創(chuàng)建一個(gè)函數(shù),然后創(chuàng)建一個(gè)線(xiàn)程對(duì)象去掉用這個(gè)函數(shù),實(shí)現(xiàn)了函數(shù)在線(xiàn)程里執(zhí)行,不影響主線(xiàn)程的執(zhí)行.
-
方法2:寫(xiě)一個(gè)自己的線(xiàn)程類(lèi)
- 1.寫(xiě)一個(gè)類(lèi),繼承自Thread類(lèi)
- 2.重寫(xiě)run方法,在里面規(guī)定需要在子線(xiàn)程中執(zhí)行的任務(wù)
- 3.在子線(xiàn)程中執(zhí)行的任務(wù)對(duì)應(yīng)的功能,如果需要參數(shù),通過(guò)對(duì)象屬性來(lái)傳值.
import requests
from threading import Thread
import re
# 下載數(shù)據(jù)
class DownloadThread(Thread):
"""下載類(lèi)"""
def __init__(self, file_path):
super().__init__()
self.file_path = file_path
def run(self):
print('開(kāi)始下載')
response = requests.request('GET', self.file_path)
data = response.content
# 獲取文件后綴名
suffix = re.search(r'\.\w+$', self.file_path).group()
with open('./abc'+suffix, 'wb') as f:
f.write(data)
print('下載完成')
if __name__ == '__main__':
print('=====')
t1 = DownloadThread('https://www.duba.com/static/images/public/20180808/news/6609c93d70cf3bc799695627dd00baa1cc112a8b.jpg')
t1.start()
print('!!!!')
#運(yùn)行
=====
開(kāi)始下載
!!!!
下載完成
注意:
此處不能用t1.run()的方式執(zhí)行run方法里面的內(nèi)容,否則會(huì)按照常規(guī)的對(duì)象方法的方式執(zhí)行,會(huì)影響主線(xiàn)程的運(yùn)行.
3.join, current_thread 和 time.time
(1)join()
如果一個(gè)任務(wù)想要在另外一個(gè)子線(xiàn)程中的任務(wù)執(zhí)行完成后再執(zhí)行,就在當(dāng)前任務(wù)前用子線(xiàn)程對(duì)象調(diào)用join方法
所以join也會(huì)阻塞線(xiàn)程,阻塞到對(duì)應(yīng)的子線(xiàn)程中的任務(wù)執(zhí)行完為止
t1.start()
t1.join()
(2)current_thread()
此方法用來(lái)判斷當(dāng)前執(zhí)行內(nèi)容在哪一個(gè)線(xiàn)程里面(這個(gè)方法在threading這個(gè)庫(kù)里)
- 如果是主線(xiàn)程,則返回的內(nèi)容包括:MainThread
- 如果是子線(xiàn)程,則返回:Thread-數(shù)字(數(shù)字從1開(kāi)始)
(3)time.time()
在python有time這樣一個(gè)庫(kù)(內(nèi)置),用于獲取當(dāng)前的時(shí)間戳,用兩個(gè)不同的時(shí)間戳求差,可算出程序執(zhí)行花費(fèi)了多少時(shí)間.
import time
start_time = time.time()
end_time = time.time()
all_time = start_time - end_time
實(shí)例:
"""__author__ = Sun Jiankang"""
from threading import Thread, current_thread
import time
from random import randint
class Download(Thread):
def __init__(self, file):
super().__init__()
self.file = file
def run(self):
"""模擬下載"""
print('開(kāi)始下載%s' % self.file)
print(current_thread())
time.sleep(randint(5, 10))
print('%s 下載結(jié)束' % self.file)
if __name__ == '__main__':
start_time = time.time()
t2 = Download('最強(qiáng).mp4')
t2.start()
print('-----------')
print(current_thread())
t1 = Download('最強(qiáng)z.mp4')
t1.start()
t1.join()
end_time = time.time()
print('總共消耗時(shí)間:%.2f' % (end_time - start_time))
# 輸出
開(kāi)始下載最強(qiáng).mp4
<Download(Thread-1, started 56716)>
-----------
<_MainThread(MainThread, started 104812)>
開(kāi)始下載最強(qiáng)z.mp4
<Download(Thread-2, started 58652)>
最強(qiáng)z.mp4 下載結(jié)束
總共消耗時(shí)間:7.00
最強(qiáng).mp4 下載結(jié)束