循環服務器模型 :循環接收客戶端請求,處理請求。同一時刻只能處理一個請求,處理完畢後再數據庫
處理下一個。服務器
IO併發模型:利用IO多路複用,異步IO等技術,同時處理多個客戶端IO請求。網絡
多進程/線程網絡併發模型:每當一個客戶端鏈接服務器,就建立一個新的進程/線程爲該客戶端服務,客戶端退出時再銷燬該進程/線程。多線程
from socket import * import os import signal # 建立監聽套接字 HOST = '0.0.0.0' PORT = 8888 ADDR = (HOST,PORT) # 客戶端服務函數 def handle(c): while True: data = c.recv(1024) if not data: break print(data.decode()) c.send(b'OK') c.close() s = socket() # tcp套接字 s.setsockopt(SOL_SOCKET,SO_REUSEADDR,1) # 設置套接字端口重用 s.bind(ADDR) s.listen(3) signal.signal(signal.SIGCHLD,signal.SIG_IGN) # 處理殭屍進程 print("Listen the port %d..." % PORT) # 循環等待客戶端鏈接 while True: try: c,addr = s.accept() except KeyboardInterrupt: os._exit(0) except Exception as e: print(e) continue # 建立子進程處理這個客戶端 pid = os.fork() if pid == 0: # 處理客戶端請求 s.close() handle(c) os._exit(0) # handle處理完客戶端請求子進程也退出 # 不管出錯或者父進程都要循環回去接受請求 # c對於父進程沒用 c.close()
from socket import * from threading import Thread import sys # 建立監聽套接字 HOST = '0.0.0.0' PORT = 8888 ADDR = (HOST,PORT) # 處理客戶端請求 def handle(c): while True: data = c.recv(1024) if not data: break print(data.decode()) c.send(b'OK') c.close() s = socket() # tcp套接字 s.setsockopt(SOL_SOCKET,SO_REUSEADDR,1) s.bind(ADDR) s.listen(3) print("Listen the port %d..."%PORT) # 循環等待客戶端鏈接 while True: try: c,addr = s.accept() except KeyboardInterrupt: sys.exit("服務器退出") except Exception as e: print(e) continue # 建立線程處理客戶端請求 t = Thread(target=handle, args=(c,)) t.setDaemon(True) # 父進程結束則全部進程終止 t.start()
項目功能 :併發
* 客戶端有簡單的頁面命令提示: 功能包含:app
* 服務器需求 :異步
技術分析:socket
粘包的處理tcp
總體結構設計ide
服務端負責接受請求,邏輯處理
from socket import *
from threading import Thread import os import time # 全局變量
HOST = '0.0.0.0' PORT = 8080 ADDR = (HOST,PORT) FTP = "/home/tarena/FTP/" # 文件庫位置
# 建立文件服務器服務端功能類
class FTPServer(Thread): def __init__(self,connfd): self.connfd = connfd super().__init__() def do_list(self): # 獲取文件列表
files = os.listdir(FTP) if not files: self.connfd.send("文件庫爲空".encode()) return
else: self.connfd.send(b'OK') time.sleep(0.1) # 防止和後面發送內容粘包
# 拼接文件列表
files_ = ""
for file in files: if file[0] != '.' and \ os.path.isfile(FTP+file): files_ += file + '\n' self.connfd.send(files_.encode()) def do_get(self,filename): try: fd = open(FTP+filename,'rb') except Exception: self.connfd.send("文件不存在".encode()) return
else: self.connfd.send(b'OK') time.sleep(0.1) # 文件發送
while True: data = fd.read(1024) if not data: time.sleep(0.1) self.connfd.send(b'##') break self.connfd.send(data) # 循環接收客戶端請求
def run(self): while True: data = self.connfd.recv(1024).decode() if not data or data == 'Q': return
elif data == 'L': self.do_list() elif data[0] == 'G': # G filename
filename = data.split(' ')[-1] self.do_get(filename) # 網絡搭建
def main(): # 建立套接字
sockfd = socket() sockfd.setsockopt(SOL_SOCKET,SO_REUSEADDR,1) sockfd.bind(ADDR) sockfd.listen(3) print("Listen the port %d..."%PORT) while True: try: connfd,addr = sockfd.accept() print("Connect from",addr) except KeyboardInterrupt: print("服務器程序退出") return
except Exception as e: print(e) continue
# 建立新的線程處理客戶端
client = FTPServer(connfd) client.setDaemon(True) client.start() # 運行run方法
if __name__ == "__main__": main()
from socket import *
import sys ADDR = ('127.0.0.1',8080) # 服務器地址
# 客戶端功能處理類
class FTPClient: def __init__(self,sockfd): self.sockfd = sockfd def do_list(self): self.sockfd.send(b'L') # 發送請求
# 等待回覆
data = self.sockfd.recv(128).decode() if data == 'OK': # 一次接收文件列表字符串
data = self.sockfd.recv(4096) print(data.decode()) else: print(data) def do_get(self,filename): # 發送請求
self.sockfd.send(('G '+filename).encode()) # 等待回覆
data = self.sockfd.recv(128).decode() if data == 'OK': fd = open(filename,'wb') # 接收文件
while True: data = self.sockfd.recv(1024) if data == b'##': break fd.write(data) fd.close() else: print(data) def do_quit(self): self.sockfd.send(b'Q') self.sockfd.close() sys.exit("謝謝使用") # 建立客戶端網絡
def main(): sockfd = socket() try: sockfd.connect(ADDR) except Exception as e: print(e) return ftp = FTPClient(sockfd) # 實例化對象
# 循環發送請求
while True: print("\n=========命令選項==========") print("**** list ****") print("**** get file ****") print("**** put file ****") print("**** quit ****") print("=============================") cmd = input("輸入命令:") if cmd.strip() == 'list': ftp.do_list() elif cmd[:3] == 'get': # get filename
filename = cmd.strip().split(' ')[-1] ftp.do_get(filename) elif cmd[:3] == 'put': # put ../filename
filename = cmd.strip().split(' ')[-1] ftp.do_put(filename) elif cmd.strip() == 'quit': ftp.do_quit() else: print("請輸入正確命令") if __name__ == "__main__": main()
定義:在內存中數據交換的操做被定義爲IO操做,IO------輸入輸出
內存和磁盤進行數據交換: 文件的讀寫 數據庫更新
內存和終端數據交換 : input print
sys.stdin sys.stdout sys.stderr
內存和網絡數據的交換: 網絡鏈接 recv send recvfrom
IO密集型程序 : 程序執行中有大量的IO操做,而較少的cpu運算操做。消耗cpu較少,IO運行時間長
CPU(計算)密集型程序:程序中存在大量的cpu運算,IO操做相對較少,消耗cpu大。
IO分爲:阻塞IO、非阻塞IO、IO多路複用、事件驅動IO、異步IO
阻塞IO
阻塞狀況:
非阻塞IO
定義 : 經過修改IO屬性行爲, 使本來阻塞的IO變爲非阻塞的狀態。
定義 :經過一個監測,能夠同時監控多個IO事件的行爲。當哪一個IO事件能夠執行,即讓這個IO事件發生。
rs, ws, xs = select(rlist, wlist, xlist[, timeout]) 監控IO事件,阻塞等待監控的IO時間發生
參數 :
返回值:
select 實現tcp服務
from socket import * from select import select # 建立一個監聽套接字做爲關注的IO s = socket() s.setsockopt(SOL_SOCKET,SO_REUSEADDR,1) s.bind(('0.0.0.0',8888)) s.listen(3) # 設置關注列表 rlist = [s] wlist = [] xlist = [s] # 循環監控IO while True: rs,ws,xs = select(rlist,wlist,xlist) # 遍歷三個返回列表,處理IO for r in rs: # 根據遍歷到IO的不一樣使用if分狀況處理 if r is s: c,addr = r.accept() print("Connect from",addr) rlist.append(c) # 增長新的IO事件 # else爲客戶端套接字就緒狀況 else: data = r.recv(1024) # 客戶端退出 if not data: rlist.remove(r) # 從關注列表移除 r.close() continue # 繼續處理其餘就緒IO print("Receive:",data.decode()) # r.send(b'OK') # 咱們但願主動處理這個IO對象 wlist.append(r) for w in ws: w.send(b'OK') wlist.remove(w) # 使用後移除 for x in xs: pass
注意:
將整數轉換爲二進制, 按照二進制位進行運算符操做
& 按位與 | 按位或 ^ 按位異或 << 左移 >> 右移
11 1011 14 1110
(11 & 14 1010) (11 | 14 1111) (11 ^ 14 0101 )
11 << 2 ===> 44 右側補0 14 >> 2 ===> 3 擠掉右側的數字
使用 : 1. 在作底層硬件時操做寄存器
2. 作標誌位的過濾
p = select.poll() 建立poll對象
p.register(fd,event) 註冊關注的IO事件
經常使用類型:
e.g. p.register(sockfd,POLLIN|POLLERR)
p.unregister(fd) 取消對IO的關注
events = p.poll()
events 是一個列表 [(fileno,evnet),(),()....]
每一個元組爲一個就緒IO,元組第一項是該IO的fileno,第二項爲該IO就緒的事件類型
poll_server 步驟
from socket import * from select import * # 建立套接字 s = socket() s.setsockopt(SOL_SOCKET,SO_REUSEADDR,1) s.bind(('0.0.0.0',8888)) s.listen(3) # 建立poll對象關注s p = poll() # 創建查找字典,用於經過fileno查找IO對象 fdmap = {s.fileno():s} # 關注s p.register(s,POLLIN|POLLERR) # 循環監控 while True: events = p.poll() # 循環遍歷發生的事件 fd-->fileno for fd,event in events: # 區分事件進行處理 if fd == s.fileno(): c,addr = fdmap[fd].accept() print("Connect from",addr) # 添加新的關注IO p.register(c,POLLIN|POLLERR) fdmap[c.fileno()] = c # 維護字典 # 按位與斷定是POLLIN就緒 elif event & POLLIN: data = fdmap[fd].recv(1024) if not data: p.unregister(fd) # 取消關注 fdmap[fd].close() del fdmap[fd] # 從字典中刪除 continue print("Receive:",data.decode()) fdmap[fd].send(b'OK')
1. 使用方法 : 基本與poll相同
2. epoll特色
from socket import * from select import * # 建立套接字 s = socket() s.setsockopt(SOL_SOCKET,SO_REUSEADDR,1) s.bind(('0.0.0.0',8888)) s.listen(3) # 建立epoll對象關注s ep = epoll() # 創建查找字典,用於經過fileno查找IO對象 fdmap = {s.fileno():s} # 關注s ep.register(s,EPOLLIN|EPOLLERR) # 循環監控 while True: events = ep.poll() # 循環遍歷發生的事件 fd-->fileno for fd,event in events: print("親,你有IO須要處理哦") # 區分事件進行處理 if fd == s.fileno(): c,addr = fdmap[fd].accept() print("Connect from",addr) # 添加新的關注IO # 將觸發方式變爲邊緣觸發 ep.register(c,EPOLLIN|EPOLLERR|EPOLLET) fdmap[c.fileno()] = c # 維護字典 # 按位與斷定是EPOLLIN就緒 # elif event & EPOLLIN: # data = fdmap[fd].recv(1024) # if not data: # ep.unregister(fd) # 取消關注 # fdmap[fd].close() # del fdmap[fd] # 從字典中刪除 # continue # print("Receive:",data.decode()) # fdmap[fd].send(b'OK')