環境:win7+python/205746.html">python/78037.html">python2.7
一直想學習多進程或多線程,但之前只是單純看一點基礎知識還有簡單的介紹,無法理解怎么去應用,直到前段時間看了github的一個爬蟲項目涉及到多進程,多線程相關內容,一邊看一邊百度相關知識點,現在把一些相關知識點和一些應用寫下來做個記錄.
首先說下什么是進程:進程是程序在計算機上的一次執行活動,當運行一個程序的時候,就啟動了一個進程.而進程又分為系統進程和用戶進程.只要是用于完成操作系統的各種功能的進程就是系統進程,它們就是處于運行狀態下的操作系統本身;而所有由你啟動的進程都是用戶進程。進程是操作系統進行資源分配的單位。
直觀點說,在任務管理器的用戶名上標明system的是系統進程,標明administrator的是用戶進程,另外net是網洛,lcacal service是本地服務,關于進程更加具體的信息可以百科,這里得省點力氣,不然收不回了.
一.多進程的簡單使用
如圖,multiprocessing有多個函數,很多我也還沒去了解,這里只講我目前了解的.

進程創建:Process(target=主要運行的函數,name=自定義進程名稱可不寫,args=(參數))
方法:
其中,Process以start()啟動某個進程。
屬性:
1.Process(),start(),join()
# -*- coding:utf-8 -*-from multiprocessing import Processimport timedef fun1(t): print 'this is fun1',time.ctime() time.sleep(t) print 'fun1 finish',time.ctime()def fun2(t): print 'this is fun2',time.ctime() time.sleep(t) print 'fun2 finish',time.ctime()if __name__ == '__main__': a=time.time() p1=Process(target=fun1,args=(4,)) p2 = Process(target=fun2, args=(6,)) p1.start() p2.start() p1.join() p2.join() b=time.time() print 'finish',b-a
這里一共開了兩個進程,p1和p2,arg=(4,)中的4是fun1函數的參數,這里要用tulpe類型,如果兩個參數或更多就是arg=(參數1,參數2...),之后用start()啟動進程,我們設置等待p1和p2進程結束再執行下一步.來看下面的運行結果,fun2和fun1基本在同一時間開始運行,當運行完畢(fun1睡眠4秒,同時fun2睡眠6秒),才執行print 'finish',b-a語句
this is fun2 Mon Jun 05 13:48:04 2017this is fun1 Mon Jun 05 13:48:04 2017fun1 finish Mon Jun 05 13:48:08 2017fun2 finish Mon Jun 05 13:48:10 2017finish 6.20300006866Process finished with exit code 0
我們再來看下start()與join()處于不同位置會發生什么
# -*- coding:utf-8 -*-from multiprocessing import Processimport timedef fun1(t): print 'this is fun1',time.ctime() time.sleep(t) print 'fun1 finish',time.ctime()def fun2(t): print 'this is fun2',time.ctime() time.sleep(t) print 'fun2 finish',time.ctime()if __name__ == '__main__': a=time.time() p1=Process(target=fun1,args=(4,)) p2 = Process(target=fun2, args=(6,)) p1.start() p1.join() p2.start() p2.join() b=time.time() print 'finish',b-a
結果:
this is fun1 Mon Jun 05 14:19:28 2017fun1 finish Mon Jun 05 14:19:32 2017this is fun2 Mon Jun 05 14:19:32 2017fun2 finish Mon Jun 05 14:19:38 2017finish 10.1229999065Process finished with exit code 0
看,現在是先運行fun1函數,運行完畢再運行fun2接著再是print 'finish',即先運行進程p1再運行進程p2,感受到join()的魅力了吧.現在再試試注釋掉join()看看又會出現什么
# -*- coding:utf-8 -*-from multiprocessing import Processimport timedef fun1(t): print 'this is fun1',time.ctime() time.sleep(t) print 'fun1 finish',time.ctime()def fun2(t): print 'this is fun2',time.ctime() time.sleep(t) print 'fun2 finish',time.ctime()if __name__ == '__main__': a=time.time() p1=Process(target=fun1,args=(4,)) p2 = Process(target=fun2, args=(6,)) p1.start() p2.start() p1.join() #p2.join() b=time.time() print 'finish',b-a
結果:
this is fun1 Mon Jun 05 14:23:57 2017this is fun2 Mon Jun 05 14:23:58 2017fun1 finish Mon Jun 05 14:24:01 2017finish 4.05900001526fun2 finish Mon Jun 05 14:24:04 2017Process finished with exit code 0
這次是運行完fun1(因為p1進程有用join(),所以主程序等待p1運行完接著執行下一步),接著繼續運行主進程的print 'finish',最后fun2運行完畢才結束
2.name,daemon,is_alive():
# -*- coding:utf-8 -*-from multiprocessing import Processimport timedef fun1(t): print 'this is fun1',time.ctime() time.sleep(t) print 'fun1 finish',time.ctime()def fun2(t): print 'this is fun2',time.ctime() time.sleep(t) print 'fun2 finish',time.ctime()if __name__ == '__main__': a=time.time() p1=Process(name='fun1進程',target=fun1,args=(4,)) p2 = Process(name='fun2進程',target=fun2, args=(6,)) p1.daemon=True p2.daemon = True p1.start() p2.start() p1.join() print p1,p2 print '進程1:',p1.is_alive(),'進程2:',p2.is_alive() #p2.join() b=time.time() print 'finish',b-a
結果:
this is fun2 Mon Jun 05 14:43:49 2017this is fun1 Mon Jun 05 14:43:49 2017fun1 finish Mon Jun 05 14:43:53 2017<Process(fun1進程, stopped daemon)> <Process(fun2進程, started daemon)>進程1: False 進程2: Truefinish 4.06500005722Process finished with exit code 0
可以看到,name是給進程賦予名字, 運行到print '進程1:',p1.is_alive(),'進程2:',p2.is_alive() 這句的時候,p1進程已經結束(返回False),p2進程仍然在運行(返回True),但p2沒有用join(),所以直接接著執行主進程,由于用了daemon=Ture,父進程終止后自動終止,p2進程沒有結束就強行結束整個程序了.
3.run()
run()在Process沒有指定target函數時,默認用run()函數運行程序,
# -*- coding:utf-8 -*-from multiprocessing import Processimport timedef fun1(t): print 'this is fun1',time.ctime() time.sleep(t) print 'fun1 finish',time.ctime()def fun2(t): print 'this is fun2',time.ctime() time.sleep(t) print 'fun2 finish',time.ctime()if __name__ == '__main__': a = time.time() p=Process() p.start() p.join() b = time.time() print 'finish', b - a
結果:
finish 0.0840001106262
從結果看出,進程p什么也沒做,為了讓進程正常運行,我們醬紫寫:
目標函數沒有參數:
# -*- coding:utf-8 -*-from multiprocessing import Processimport timedef fun1(): print 'this is fun1',time.ctime() time.sleep(2) print 'fun1 finish',time.ctime()def fun2(t): print 'this is fun2',time.ctime() time.sleep(t) print 'fun2 finish',time.ctime()if __name__ == '__main__': a = time.time() p=Process() p.run=fun1 p.start() p.join() b = time.time() print 'finish', b - a
結果:
this is fun1 Mon Jun 05 16:34:41 2017fun1 finish Mon Jun 05 16:34:43 2017finish 2.11500000954Process finished with exit code 0
目標函數有參數:
# -*- coding:utf-8 -*-from multiprocessing import Processimport timedef fun1(t): print 'this is fun1',time.ctime() time.sleep(t) print 'fun1 finish',time.ctime()def fun2(t): print 'this is fun2',time.ctime() time.sleep(t) print 'fun2 finish',time.ctime()if __name__ == '__main__': a = time.time() p=Process() p.run=fun1(2) p.start() p.join() b = time.time() print 'finish', b - a
結果:
this is fun1 Mon Jun 05 16:36:27 2017fun1 finish Mon Jun 05 16:36:29 2017Process Process-1:Traceback (most recent call last): File "E:/Anaconda2/lib/multiprocessing/process.py", line 258, in _bootstrap self.run()TypeError: 'NoneType' object is not callablefinish 2.0529999733Process finished with exit code 0
目標函數有參數的出現了異常,為什么呢?我現在還找不到原因,但是實踐發現,當最后一個參數賦予進程運行后,沒有其他參數,就會出現這個異常,有人知道的望告知.
二.進程池
對于需要使用幾個甚至十幾個進程時,我們使用Process還是比較方便的,但是如果要成百上千個進程,用Process顯然太笨了,multiprocessing提供了Pool類,即現在要講的進程池,能夠將眾多進程放在一起,設置一個運行進程上限,每次只運行設置的進程數,等有進程結束,再添加新的進程
Pool(processes =num):設置運行進程數,當一個進程運行完,會添加新的進程進去
apply_async(函數,(參數)):非阻塞,其中參數是tulpe類型,
apply(函數,(參數)):阻塞
close():關閉pool,不能再添加新的任務
terminate():結束運行的進程,不再處理未完成的任務
join():和Process介紹的作用一樣, 但要在close或terminate之后使用。
1.單個進程池
# -*- coding:utf-8 -*-from multiprocessing import Poolimport timedef fun1(t): print 'this is fun1',time.ctime() time.sleep(t) print 'fun1 finish',time.ctime()def fun2(t): print 'this is fun2',time.ctime() time.sleep(t) print 'fun2 finish',time.ctime()if __name__ == '__main__': a=time.time() pool = Pool(processes =3) # 可以同時跑3個進程 for i in range(3,8): pool.apply_async(fun1,(i,)) pool.close() pool.join() b=time.time() print 'finish',b-a
結果:
this is fun1 Mon Jun 05 15:15:38 2017this is fun1 Mon Jun 05 15:15:38 2017this is fun1 Mon Jun 05 15:15:38 2017fun1 finish Mon Jun 05 15:15:41 2017this is fun1 Mon Jun 05 15:15:41 2017fun1 finish Mon Jun 05 15:15:42 2017this is fun1 Mon Jun 05 15:15:42 2017fun1 finish Mon Jun 05 15:15:43 2017fun1 finish Mon Jun 05 15:15:47 2017fun1 finish Mon Jun 05 15:15:49 2017finish 11.1370000839Process finished with exit code 0
從上面的結果可以看到,設置了3個運行進程上限,15:15:38這個時間同時開始三個進程,當第一個進程結束時(參數為3秒那個進程),會添加新的進程,如此循環,直至進程池運行完再執行主進程語句b=time.time() print 'finish',b-a .這里用到非阻塞apply_async(),再來對比下阻塞apply()
# -*- coding:utf-8 -*-from multiprocessing import Poolimport timedef fun1(t): print 'this is fun1',time.ctime() time.sleep(t) print 'fun1 finish',time.ctime()def fun2(t): print 'this is fun2',time.ctime() time.sleep(t) print 'fun2 finish',time.ctime()if __name__ == '__main__': a=time.time() pool = Pool(processes =3) # 可以同時跑3個進程 for i in range(3,8): pool.apply(fun1,(i,)) pool.close() pool.join() b=time.time() print 'finish',b-a
結果:
this is fun1 Mon Jun 05 15:59:26 2017fun1 finish Mon Jun 05 15:59:29 2017this is fun1 Mon Jun 05 15:59:29 2017fun1 finish Mon Jun 05 15:59:33 2017this is fun1 Mon Jun 05 15:59:33 2017fun1 finish Mon Jun 05 15:59:38 2017this is fun1 Mon Jun 05 15:59:38 2017fun1 finish Mon Jun 05 15:59:44 2017this is fun1 Mon Jun 05 15:59:44 2017fun1 finish Mon Jun 05 15:59:51 2017finish 25.1610000134Process finished with exit code 0
可以看到,阻塞是當一個進程結束后,再進行下一個進程,一般我們都用非阻塞apply_async()
2.多個進程池
上面是使用單個進程池的,對于多個進程池,我們可以用for循環,直接看代碼
# -*- coding:utf-8 -*-from multiprocessing import Poolimport timedef fun1(t): print 'this is fun1',time.ctime() time.sleep(t) print 'fun1 finish',time.ctime()def fun2(t): print 'this is fun2',time.ctime() time.sleep(t) print 'fun2 finish',time.ctime()if __name__ == '__main__': a=time.time() pool = Pool(processes =3) # 可以同時跑3個進程 for fun in [fun1,fun2]: for i in range(3,8): pool.apply_async(fun,(i,)) pool.close() pool.join() b=time.time() print 'finish',b-a
結果:
this is fun1 Mon Jun 05 16:04:38 2017this is fun1 Mon Jun 05 16:04:38 2017this is fun1 Mon Jun 05 16:04:38 2017fun1 finish Mon Jun 05 16:04:41 2017this is fun1 Mon Jun 05 16:04:41 2017fun1 finish Mon Jun 05 16:04:42 2017this is fun1 Mon Jun 05 16:04:42 2017fun1 finish Mon Jun 05 16:04:43 2017this is fun2 Mon Jun 05 16:04:43 2017fun2 finish Mon Jun 05 16:04:46 2017this is fun2 Mon Jun 05 16:04:46 2017fun1 finish Mon Jun 05 16:04:47 2017this is fun2 Mon Jun 05 16:04:47 2017fun1 finish Mon Jun 05 16:04:49 2017this is fun2 Mon Jun 05 16:04:49 2017fun2 finish Mon Jun 05 16:04:50 2017this is fun2 Mon Jun 05 16:04:50 2017fun2 finish Mon Jun 05 16:04:52 2017fun2 finish Mon Jun 05 16:04:55 2017fun2 finish Mon Jun 05 16:04:57 2017finish 19.1670000553Process finished with exit code 0
看到了,在fun1運行完接著運行fun2.
另外對于沒有參數的情況,就直接 pool.apply_async(funtion),無需寫上參數.
在學習編寫程序過程,曾遇到不用if _name_ == '_main_':而直接運行程序,這樣結果會出錯,經查詢,在Windows上要想使用進程模塊,就必須把有關進程的代碼寫在當前.py文件的if _name_ == ‘_main_' :語句的下面,才能正常使用Windows下的進程模塊。Unix/Linux下則不需要。原因有人這么說:在執行的時候,由于你寫的 py 會被當成module 讀進執行。所以,一定要判斷自身是否為 _main_。也就是要:
if __name__ == ‘__main__' :# do something.
這里我自己還搞不清楚,期待以后能夠理解
學習的過程中,還涉及了經常和進程一起運用的隊列Queue和線程threading,有時間以后再寫吧,希望對大家的學習有所幫助,也希望大家多多支持VEVB武林網。
新聞熱點
疑難解答