python 開發(fā)的三種運行模式詳細(xì)介紹
Python 三種運行模式
Python作為一門腳本語言,使用的范圍很廣。有的同學(xué)用來算法開發(fā),有的用來驗證邏輯,還有的作為膠水語言,用它來粘合整個系統(tǒng)的流程。不管怎么說,怎么使用python既取決于你自己的業(yè)務(wù)場景,也取決于你自己的python應(yīng)用能力。就我個人而言,我覺得python作為既可以用來進(jìn)行業(yè)務(wù)的開發(fā),也可以進(jìn)行產(chǎn)品原型的開發(fā).一般來說,python的運行主要下面這三種模式。
1.單循環(huán)模式
單循環(huán)模式使用的最多,也最簡單,當(dāng)然也最穩(wěn)定。為什么呢,因為單循環(huán)本來代碼就寫的很少,出錯的機會就更少,所以一般只要寫對了接口,犯錯誤的機會還是很低的。當(dāng)然,我們不是說單循環(huán)就沒什么用,恰恰相反。單循環(huán)模式是我們最經(jīng)常使用的一種模式。這種開發(fā)對于一些小工具、小應(yīng)用、小場景特別合適。
#!/usr/bin/python
import os
import sys
import re
import signal
import time
g_exit = 0
def sig_process(sig, frame):
global g_exit
g_exit = 1
print 'catch signal'
def main():
global g_exit
signal.signal(signal.SIGINT, sig_process)
while 0 == g_exit:
time.sleep(1)
'''
module process code
'''
if __name__ == '__main__':
main()
2.多線程模式
多線程模式經(jīng)常用在那些容易阻塞的場合。比如多線程客戶端讀寫,多線程web訪問等等。這里的多線程有個特點,那就是每個線程都是按照客戶端創(chuàng)建的。簡單的舉例就是服務(wù)器socket,來一個socket創(chuàng)建一個thread,這樣如果存在多個用戶的話,就有多個thread并發(fā)連接。這種方式比較簡單,用起來很快,缺點就是所有業(yè)務(wù)有可能并發(fā)執(zhí)行,全局?jǐn)?shù)據(jù)保護起來很麻煩。
#!/usr/bin/python
import os
import sys
import re
import signal
import time
import threading
g_exit=0
def run_thread():
global g_exit
while 0 == g_exit:
time.sleep(1)
'''
do jobs per thread
'''
def sig_process(sig, frame):
global g_exit
g_exit = 1
def main():
global g_exit
signal.signal(signal.SIGINT, sig_process)
g_threads = []
for i in range(4):
td = threading.Thread(target = run_thread)
td.start()
g_threads.append(td)
while 0 == g_exit:
time.sleep(1)
for i in range(4):
g_threads[i].join()
if __name__ == '__main__':
main()
3.reactor模式
reactor模式,不復(fù)雜,簡單的來說,就是利用多線程來處理每一個業(yè)務(wù)。如果一個業(yè)務(wù)已經(jīng)被某一個thread處理了,那么其他的thread就不能再次處理這個業(yè)務(wù)了。這樣,它相當(dāng)于解決了一個問題,也就是我們在前面所說的鎖的問題。因此,對于這種模式的開發(fā)者來說,編寫業(yè)務(wù)其實是一件簡單的事情,因為他所要關(guān)注的只是自己的一畝三分地就可以了。之前云風(fēng)同學(xué)編寫的skynet就是這么一種模式,只不過它使用了c+lua來開發(fā)的。其實只要了解了reactor模式本身,用什么語言開發(fā)不重要,關(guān)鍵是理解reactor的精髓就可以了。

如果寫成code,那應(yīng)該是這樣的,
#!/usr/bin/python
import os
import sys
import re
import time
import signal
import threading
g_num = 4
g_exit =0
g_threads = []
g_sem = []
g_lock = threading.Lock()
g_event = {}
def add_event(name, data):
global g_lock
global g_event
if '' == name:
return
g_lock.acquire()
if name in g_event:
g_event[name].append(data)
g_lock.release()
return
g_event[name] = []
'''
0 means idle, 1 means busy
'''
g_event[name].append(0)
g_event[name].append(data)
g_lock.release()
def get_event(name):
global g_lock
global g_event
g_lock.acquire()
if '' != name:
if [] != g_event[name]:
if 1 != len(g_event[name]):
data = g_event[name][1]
del g_event[name][1]
g_lock.release()
return name, data
else:
g_event[name][0] = 0
for k in g_event:
if 1 == len(g_event[k]):
continue
if 1 == g_event[k][0]:
continue
g_event[k][0] =1
data = g_event[k][1]
del g_event[k][1]
g_lock.release()
return k, data
g_lock.release()
return '', -1
def sig_process(sig, frame):
global g_exit
g_exit =1
print 'catch signal'
def run_thread(num):
global g_exit
global g_sem
global g_lock
name = ''
data = -1
while 0 == g_exit:
g_sem[num].acquire()
while True:
name, data = get_event(name)
if '' == name:
break
g_lock.acquire()
print name, data
g_lock.release()
def test_thread():
global g_exit
while 0 == g_exit:
for i in range(100):
add_event('1', (i << 2) + 0)
add_event('2', (i << 2) + 1)
add_event('3', (i << 2) + 2)
add_event('4', (i << 2) + 3)
time.sleep(1)
def main():
global g_exit
global g_num
global g_threads
global g_sem
signal.signal(signal.SIGINT, sig_process)
for i in range(g_num):
sem = threading.Semaphore(0)
g_sem.append(sem)
td = threading.Thread(target=run_thread, args=(i,))
td.start()
g_threads.append(td)
'''
test thread to give data
'''
test = threading.Thread(target=test_thread)
test.start()
while 0 == g_exit:
for i in range(g_num):
g_sem[i].release()
time.sleep(1)
'''
call all thread to close
'''
for i in range(g_num):
g_sem[i].release()
for i in range(g_num):
g_threads[i].join()
test.join()
print 'exit now'
'''
entry
'''
if __name__ == '__main__':
main()
感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
- Python的Flask框架及Nginx實現(xiàn)靜態(tài)文件訪問限制功能
- Python的Bottle框架中返回靜態(tài)文件和JSON對象的方法
- 使用nodejs、Python寫的一個簡易HTTP靜態(tài)文件服務(wù)器
- python中實現(xiàn)迭代器(iterator)的方法示例
- Python中二維列表如何獲取子區(qū)域元素的組成
- python解決漢字編碼問題:Unicode Decode Error
- Python 3.x 連接數(shù)據(jù)庫示例(pymysql 方式)
- 一步步教你用Python實現(xiàn)2048小游戲
- python django 訪問靜態(tài)文件出現(xiàn)404或500錯誤
- Python使用中文正則表達(dá)式匹配指定中文字符串的方法示例
相關(guān)文章
python json.loads兼容單引號數(shù)據(jù)的方法
今天小編就為大家分享一篇python json.loads兼容單引號數(shù)據(jù)的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-12-12
Anaconda和ipython環(huán)境適配的實現(xiàn)
這篇文章主要介紹了Anaconda和ipython環(huán)境適配的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04
python中的循環(huán)結(jié)構(gòu)問題
這篇文章主要介紹了python中的循環(huán)結(jié)構(gòu)問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-03-03
淺談python裝飾器探究與參數(shù)的領(lǐng)取
下面小編就為大家分享一篇淺談python裝飾器探究與參數(shù)的領(lǐng)取,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2017-12-12
在python3環(huán)境下的Django中使用MySQL數(shù)據(jù)庫的實例
下面小編就為大家?guī)硪黄趐ython3環(huán)境下的Django中使用MySQL數(shù)據(jù)庫的實例。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-08-08
python利用urllib和urllib2訪問http的GET/POST詳解
urllib模塊提供的上層接口,使我們可以像讀取本地文件一樣讀取www和ftp上的數(shù)據(jù)。下面這篇文章主要給大家介紹了關(guān)于python如何利用urllib和urllib2訪問http的GET/POST的相關(guān)資料,需要的朋友可以參考借鑒,下面來一起看看吧。2017-09-09
django restframework序列化字段校驗規(guī)則
本文主要介紹了django restframework序列化字段校驗規(guī)則,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05

