Python 多進程多線編程模板

来源:http://www.cnblogs.com/scorpion-ph/archive/2017/05/12/6846803.html
-Advertisement-
Play Games

Python 多進程多線程原理介紹以及原創的Python多進程和多線程模板 ...


一、Python 多進程多線程原理介紹

  1. Python 全局解釋器鎖GIL

    a) Python的全局解釋器鎖GIL是互斥鎖,能夠防止本機多個線程一次執行Python位元組碼;由於CPython的記憶體管理線上程級別是不安全的(記憶體泄露),所以這個全局解釋器鎖是必須的。每個Python進程只能申請使用一個GIL鎖,因此Python的多線程雖然是併發的但不能並行處理。Python的解釋器每次只能執行一個線程,待GIL鎖釋放後再執行下一個線程,這樣線程輪流被執行。

   b) Python2.x里,GIL的釋放邏輯是當前線程遇見IO操作或者ticks計數達到100;python3.x中,GIL不使用ticks計數,改為使用計時器即執行時間達到設定閾值後,釋放當前線程的GIL。

  2. Python 多進程多線程適用場景

    a) CPU-bound (計算密集型

      CPU-bound 指的是系統的硬碟/記憶體效能相對CPU的效能要好很多,系統運行時的狀態是CPU占用接近100%;I/O讀寫硬碟和記憶體占用很低。在這種計算密集型的狀況下Python的ticks計數器很快達到閾值會觸發GIL的釋放與再鎖定。這時Python多線程會頻繁的加鎖和釋放鎖,消耗大量的CPU資源。因此對於計算密集型的程式Python使用多進程要比多線程好很多

    b) I/O-bound (I/O密集型

      I/O-bound 指的是系統的CPU效能相對硬碟/記憶體的效能要好很多,系統運行時大部分時間是在等待I/O讀寫,CPU的占用並不高。Python單線程下有I/O操作會進行I/O等待,造成不必要的CPU時間浪費,開啟多線程能線上程A執行I/O等待時,自動切換到線程B,可以不浪費CPU的資源,從而能提升程式執行效率。因此對於I/O密集型的程式Python多線程有一定的優勢。

二、 Python 多進程多線程編程模板

  1. 此模板具有兩個重要優點:

   a) 根據程式執行類型自定義多進程多線程的開啟個數,調節程式的執行效率。

   b) 程式中還實時列印程式執行進度並不較大的影響程式的執行效率。

 

  1 #!/usr/bin/env python
  2 # -*- coding: UTF-8 -*-
  3 __author__="阿輝楓情"
  4 __date__ = "$2017-5-12 21:49:51$"
  5 import os
  6 import sys
  7 import time
  8 import random
  9 import threading
 10 from multiprocessing import Process, Manager, Lock
 11 
 12 # 解決編碼問題
 13 reload(sys)
 14 sys.setdefaultencoding('utf-8')
 15 Type = sys.getfilesystemencoding()
 16 
 17 #------------------------------------------------
 18 #   可修改的全局變數參數--Start.
 19 tasklist = [] # 定義任務列表
 20 
 21 PROCESS_COUNT = 2 # 定義進程數量
 22 THREAD_COUNT = 4 # 定義線程數量
 23 #   可修改全局變數參數--End.
 24 #------------------------------------------------
 25 
 26 
 27 class HandleTask(threading.Thread):
 28     """docstring for HandleTask"""
 29 
 30     def __init__(self, proid, prolock, thrid, thrlock, tasklist, tasknum, schedule):
 31         super(HandleTask, self).__init__()
 32         self.proid = proid
 33         self.prolock = prolock
 34         self.thrid = thrid
 35         self.thrlock = thrlock
 36         self.tasklist = tasklist
 37         self.tasknum = tasknum
 38         self.sch = schedule
 39         self.pid = os.getpid()
 40 
 41     def run(self):
 42         self.prolock.acquire()
 43         self.thrlock.acquire()
 44         print "The Thread [%s:%s] tasklist number:[%s]" % (self.proid, self.thrid, len(self.tasklist))
 45         self.thrlock.release()
 46         self.prolock.release()
 47 
 48         for (element, ) in self.tasklist:
 49             # 任務執行開始
 50             # print element
 51             time.sleep(1)
 52             # 任務執行結束
 53 
 54             self.prolock.acquire()
 55             self.thrlock.acquire()
 56             self.sch.value += 1
 57             self.thrlock.release()
 58             self.prolock.release()
 59 
 60 def Thread_Handle(proid, prolock, tasklist, tasknum, schedule):
 61     global THREAD_COUNT
 62     lock = threading.Lock()
 63     WorksThread = []
 64     thread_task_number = len(tasklist) / THREAD_COUNT
 65     if thread_task_number == 0:
 66         THREAD_COUNT = len(tasklist)
 67         thread_task_number = 1
 68 
 69     for i in range(THREAD_COUNT):
 70         if i != THREAD_COUNT - 1:
 71             source_list = tasklist[i * thread_task_number: (i + 1) * thread_task_number]
 72         else:
 73             source_list = tasklist[i * thread_task_number:]
 74         Work = HandleTask(proid, prolock, i, lock, source_list, tasknum, schedule)
 75         Work.start()
 76         WorksThread.append(Work)
 77 
 78     for Work in WorksThread:
 79         Work.join()  
 80 
 81 def Process_Handle(tasklist, tasknum):
 82     global PROCESS_COUNT
 83     lock = Lock()
 84     # 定義進度變數 schedule
 85     schedule = Manager().Value('schedule', 0)
 86     WorksProcess = []
 87     # 按照任務大小進行進程任務分配
 88     process_task_num = len(tasklist) / PROCESS_COUNT
 89     if process_task_num == 0:
 90         PROCESS_COUNT = len(tasklist)
 91         process_task_num = 1
 92 
 93     for i in range(PROCESS_COUNT):
 94         if i != PROCESS_COUNT - 1:
 95             source_list = tasklist[i * process_task_num: (i + 1) * process_task_num]
 96         else:
 97             source_list = tasklist[i * process_task_num:]
 98         Work = Process(target=Thread_Handle, args=(i, lock, source_list, tasknum, schedule))
 99         Work.start()
100         WorksProcess.append(Work)
101     # 添加額外進程列印任務執行進度
102     Work = Process(target=Displays, args=(lock, tasknum, schedule))
103     Work.start()
104     WorksProcess.append(Work)
105     for Work in WorksProcess:
106         Work.join()
107     del WorksProcess
108 
109 def Displays(prolock, tasknum, schedule, delaytime=None):
110     if delaytime is None:
111         delaytime = 1
112     while (tasknum - schedule.value):
113         time.sleep(delaytime)
114         print "Completed:[%s] , Remaining:[%s]" % (schedule.value, tasknum - schedule.value)
115 
116 def main():
117     # 列印輸出主進程號
118     print "The Main Process ID:[%s]"% os.getpid()
119     # 建立測試任務
120     for i in range(1, 101):
121         tasklist.append((i, ))
122     Process_Handle(tasklist, len(tasklist))
123 
124 
125 if __name__ == '__main__':
126 
127     print "The Program start time:", time.strftime("%Y-%m-%d %H:%M:%S", time.localtime())
128     start = time.time()
129     main()
130     print "The Program end time:", time.strftime("%Y-%m-%d %H:%M:%S", time.localtime()), "[%s]" % (time.time() - start)
131     raw_input("Please enter any key to end!!!".decode('utf-8').encode(Type))
View Code

  2. 程式執行測試結果

 1 The Program start time: 2017-05-12 18:15:12
 2 The Main Process ID:[9752]
 3 The Thread [0:0] tasklist number:[12]
 4 The Thread [1:0] tasklist number:[12]
 5 The Thread [1:1] tasklist number:[12]
 6 The Thread [0:1] tasklist number:[12]
 7 The Thread [1:2] tasklist number:[12]
 8 The Thread [0:2] tasklist number:[12]
 9 The Thread [0:3] tasklist number:[14]
10 The Thread [1:3] tasklist number:[14]
11 Completed:[0] , Remaining:[100]
12 Completed:[8] , Remaining:[92]
13 Completed:[16] , Remaining:[84]
14 Completed:[25] , Remaining:[75]
15 Completed:[34] , Remaining:[66]
16 Completed:[42] , Remaining:[57]
17 Completed:[51] , Remaining:[49]
18 Completed:[59] , Remaining:[41]
19 Completed:[67] , Remaining:[32]
20 Completed:[76] , Remaining:[24]
21 Completed:[85] , Remaining:[15]
22 Completed:[93] , Remaining:[7]
23 Completed:[98] , Remaining:[2]
24 Completed:[100] , Remaining:[0]
25 The Program end time: 2017-05-12 18:15:27 [15.007999897]
26 Please enter any key to end!!!
View Code

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 學習 ASP .NET 時間的第三周: 來講講如何在 visual studio 2013...上搭載 SVN吧: 廢話不多說: One Step: 電腦上已安裝 visual studio 2013 等版本(未安裝時 紅色區域是不存在的) Two Step: 從官網上下載對應的visualsvnh ...
  • 常見的理解問題: const char * * s;//表示s是指向const char * 類型的指針; char * * const s;//表示s是指向char * 類型的一個常量指針。 ...
  • 今天花了一天的時間在Linux 系統上完成了歐拉計劃第13題的Larger Sum編程(花費的時間有點長啊)。既然耗了這麼長的時間,那就要好好的總結一下,不然也對不起這一天的時間。這次主要是在ubantu 上編程,簡單的設計到了編譯、調試、鏈接和簡單的編寫Makefile。 Larger Sum 中 ...
  • import java.io.FileInputStream;import java.io.FileOutputStream;import java.io.IOException;import java.util.LinkedList;import java.util.List; //文件複製 // ...
  • 問題:char[]與String相比,有什麼優勝的地方? 回答: 針對安全保密高的信息,char[]比String做得更好。因為String是不可變得,即使你修改原先的變數,實際上也是在記憶體中新建一個對象,原數據還是保留在記憶體中,等待回收。而char[]中的元素是可以更改的。這就意味著,如密碼等保密 ...
  • You are given an n x n 2D matrix representing an image. Rotate the image by 90 degrees (clockwise). Follow up: Could you do this in-place? ...
  • import java.awt.Color;import java.awt.Font;import java.awt.Graphics;import java.awt.Toolkit;import java.awt.event.MouseAdapter;import java.awt.event.M ...
  • Given a collection of distinct numbers, return all possible permutations. ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...