python 實(shí)現(xiàn)多線程的三種方法總結(jié)
import threadingimport timeimport _threaddef job(): print('這是一個(gè)需要執(zhí)行的任務(wù)。。。。。') print('當(dāng)前線程的個(gè)數(shù):', threading.active_count() ) print('當(dāng)前線程的信息:', threading.current_thread()) time.sleep(100)if __name__ == ’__main__’: # 創(chuàng)建多線程時(shí), 需要制定該線程執(zhí)行的任務(wù) _thread.start_new_thread(job, ()) _thread.start_new_thread(job, ()) job()2.threading.Thread
import threadingimport timedef job(): print('這是一個(gè)需要執(zhí)行的任務(wù)。。。。。') print('當(dāng)前線程的個(gè)數(shù):', threading.active_count() ) time.sleep(1) print('當(dāng)前線程的信息:', threading.current_thread())if __name__ == ’__main__’: # 創(chuàng)建多線程時(shí), 需要制定該線程執(zhí)行的任務(wù).name線程名字 target目標(biāo)函數(shù)名 t1 = threading.Thread(target=job,name=’job1’) t2 = threading.Thread(target=job,name=’job2’) t1.start() t2.start() print(threading.active_count()) print('程序執(zhí)行結(jié)束.....')
輸出:
這是一個(gè)需要執(zhí)行的任務(wù)。。。。。
當(dāng)前線程的個(gè)數(shù): 3
這是一個(gè)需要執(zhí)行的任務(wù)。。。。。
3
程序執(zhí)行結(jié)束.....
當(dāng)前線程的個(gè)數(shù): 3
當(dāng)前線程的信息: <Thread(job1, started 140416648140544)>
當(dāng)前線程的信息: <Thread(job2, started 140416639747840)>
出現(xiàn)的問題: 主線程執(zhí)行結(jié)束, 但是子線程還在運(yùn)行。
join()方法可以等待所有的子線程執(zhí)行結(jié)束之后, 再執(zhí)行主線程。
import threadingimport timedef job(): print('這是一個(gè)需要執(zhí)行的任務(wù)。。。。。') print('當(dāng)前線程的個(gè)數(shù):', threading.active_count() ) print('當(dāng)前線程的信息:', threading.current_thread()) time.sleep(1)if __name__ == ’__main__’: # 創(chuàng)建多線程時(shí), 需要制定該線程執(zhí)行的任務(wù).name線程名字 target目標(biāo)函數(shù)名 t1 = threading.Thread(target=job,name=’job1’) t2 = threading.Thread(target=job,name=’job2’) t1.start() t2.start() print(threading.active_count()) # 出現(xiàn)的問題: 主線程執(zhí)行結(jié)束, 但是子線程還在運(yùn)行。 # 等待所有的子線程執(zhí)行結(jié)束之后, 再執(zhí)行主線程 t1.join() t2.join() print('程序執(zhí)行結(jié)束.....')
之前寫過(guò)一個(gè)簡(jiǎn)單爬蟲的實(shí)驗(yàn),現(xiàn)在希望獲取十個(gè)ip的城市和國(guó)家
-不使用多線程import timefrom urllib.request import urlopen# 記錄時(shí)間的裝飾器def timeit(f): def wrapper(*args, **kwargs):start_time = time.time()res = f(*args, **kwargs)end_time = time.time()print('%s函數(shù)運(yùn)行時(shí)間:%.2f' % (f.__name__, end_time - start_time))return res return wrapperdef get_addr(ip): url = 'http://ip-api.com/json/%s' % (ip) urlObj = urlopen(url) # 服務(wù)端返回的頁(yè)面信息, 此處為字符串類型 pageContent = urlObj.read().decode(’utf-8’) # 2. 處理Json數(shù)據(jù) import json # 解碼: 將json數(shù)據(jù)格式解碼為python可以識(shí)別的對(duì)象; dict_data = json.loads(pageContent) print('''%s 所在城市: %s 所在國(guó)家: %s''' % (ip, dict_data[’city’], dict_data[’country’]))@timeitdef main(): ips = [’12.13.14.%s’ % (i + 1) for i in range(10)] for ip in ips:get_addr(ip)if __name__ == ’__main__’: main()
時(shí)間需要138.91秒。
-使用多線程import threadingimport timefrom urllib.request import urlopendef timeit(f): def wrapper(*args, **kwargs):start_time = time.time()res = f(*args, **kwargs)end_time = time.time()print('%s函數(shù)運(yùn)行時(shí)間:%.2f' % (f.__name__, end_time - start_time))return res return wrapperdef get_addr(ip): url = 'http://ip-api.com/json/%s' % (ip) urlObj = urlopen(url) # 服務(wù)端返回的頁(yè)面信息, 此處為字符串類型 pageContent = urlObj.read().decode(’utf-8’) # 2. 處理Json數(shù)據(jù) import json # 解碼: 將json數(shù)據(jù)格式解碼為python可以識(shí)別的對(duì)象; dict_data = json.loads(pageContent) print('''%s 所在城市: %s 所在國(guó)家: %s''' % (ip, dict_data[’city’], dict_data[’country’]))@timeitdef main(): ips = [’12.13.14.%s’ % (i + 1) for i in range(10)] threads = [] for ip in ips:# 實(shí)例化10個(gè)對(duì)象,target=目標(biāo)函數(shù)名,args=目標(biāo)函數(shù)參數(shù)(元組格式)t = threading.Thread(target=get_addr, args=(ip, ))threads.append(t)t.start()# 等待所有子線程結(jié)束再運(yùn)行主線程 [thread.join() for thread in threads]if __name__ == ’__main__’: main()
重寫run方法, 實(shí)現(xiàn)多線程, 因?yàn)閟tart方法執(zhí)行時(shí), 調(diào)用的是run方法;run方法里面編寫的內(nèi)容就是你要執(zhí)行的任務(wù);
import threadingimport time# 重寫一個(gè)類,繼承于threading.Threadclass MyThread(threading.Thread): def __init__(self, jobName):super(MyThread, self).__init__()self.jobName = jobName # 重寫run方法, 實(shí)現(xiàn)多線程, 因?yàn)閟tart方法執(zhí)行時(shí), 調(diào)用的是run方法; # run方法里面編寫的內(nèi)容就是你要執(zhí)行的任務(wù); def run(self):print('這是一個(gè)需要執(zhí)行的任務(wù)%s。。。。。' %(self.jobName))print('當(dāng)前線程的個(gè)數(shù):', threading.active_count() )time.sleep(1)print('當(dāng)前線程的信息:', threading.current_thread())if __name__ == ’__main__’: t1 = MyThread('name1') t2 = MyThread('name2') t1.start() t2.start() t1.join() t2.join() print('程序執(zhí)行結(jié)束.....')
重寫run方法實(shí)現(xiàn)剛才爬蟲多線程案例
import threadingimport timefrom urllib.request import urlopendef timeit(f): def wrapper(*args, **kwargs):start_time = time.time()res = f(*args, **kwargs)end_time = time.time()print('%s函數(shù)運(yùn)行時(shí)間:%.2f' % (f.__name__, end_time - start_time))return res return wrapperclass MyThread(threading.Thread): def __init__(self, ip):super(MyThread, self).__init__()self.ip = ip def run(self):url = 'http://ip-api.com/json/%s' % (self.ip)urlObj = urlopen(url)# 服務(wù)端返回的頁(yè)面信息, 此處為字符串類型pageContent = urlObj.read().decode(’utf-8’)# 2. 處理Json數(shù)據(jù)import json# 解碼: 將json數(shù)據(jù)格式解碼為python可以識(shí)別的對(duì)象;dict_data = json.loads(pageContent)print(''' %s所在城市: %s所在國(guó)家: %s''' % (self.ip, dict_data[’city’], dict_data[’country’]))@timeitdef main(): ips = [’12.13.14.%s’ % (i + 1) for i in range(10)] threads = [] for ip in ips:# 實(shí)例化自己重寫的類t = MyThread(ip)threads.append(t)t.start() [thread.join() for thread in threads]if __name__ == ’__main__’: main()
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持好吧啦網(wǎng)。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。
相關(guān)文章:
1. ASP.NET MVC實(shí)現(xiàn)樹形導(dǎo)航菜單2. 多級(jí)聯(lián)動(dòng)下拉選擇框,動(dòng)態(tài)獲取下一級(jí)3. 如何封裝一個(gè)Ajax函數(shù)4. jsp網(wǎng)頁(yè)實(shí)現(xiàn)貪吃蛇小游戲5. Ajax常用封裝庫(kù)——Axios的使用6. jsp+servlet簡(jiǎn)單實(shí)現(xiàn)上傳文件功能(保存目錄改進(jìn))7. CSS Hack大全-教你如何區(qū)分出IE6-IE10、FireFox、Chrome、Opera8. WML學(xué)習(xí)之七 CGI編程9. Vue實(shí)現(xiàn)用戶沒有登陸時(shí),訪問后自動(dòng)跳轉(zhuǎn)登錄頁(yè)面的實(shí)現(xiàn)思路10. 微信開發(fā) 網(wǎng)頁(yè)授權(quán)獲取用戶基本信息
