單進程、單線程、非阻塞

單進程-單線程-非阻塞-長鏈接

import socket

import re

def service_client(new_socket):

# '''為這個客戶端返回數據'''

# 1.接收瀏覽器發送出來的請求,即HTTP請求

# GET/HTTP/1.1

#

? ? request_lines = request.splotlines()

print ('')

print ('>>>' *30)

print (request_lines)

ret = re.match(r'[^/]+(/[^ ]*)', request_lines[0])#正則提取

? ? file_name =''

? ? if ret:

file_name = ret.group(1)

print ('*' *50, file_name)# !!!!!!!!網頁名字

? ? ? ? if file_name ='/index.html'

? ? # 2.返回HTTP格式的數據給瀏覽器

? ? try:

f =open('.../Desktop/項目/項目.html', 'rb')

# f = open('.../Desktop/項目/項目.html' +file_name,'rb')

? ? except:

response ='HTTP/1.1 404 NOT FOUND\r\n'

? ? ? ? response +='\r\n'

? ? ? ? response +='--FILE NOT FOUND---'

? ? ? ? new_socket.send(response.encode('utf-8'))

else:

html_content = f.read()

f.close()

# 2.1 準備發送給瀏覽器的Header

? ? ? ? response_header ='HTTP/1.1 200 OK\r\n'

? ? ? ? response_header +='\r\n'

? ? ? ? response_header ='Content-Length : %d\r\n'%len(response_body)

response_header +='\r\n'

? ? ? ? response = response_header.encode('utf-8')+ response_body

#將 Response發送給瀏覽器

? ? ? ? new_socket.send(response)

def main():

""" 用來完成整提的控制"""

? ? # 1.創建套接字

? ? tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

tcp_server_socket.setsockopt(socket.SOCK_STREAM, socket.SO_REUSEADDR,1)

# 2.綁定

? ? tcp_server_socket.bind(('', 7890))

# 3.變為監聽套接字(最大鏈接書128)

? ? tcp_server_socket.listen(128)

while True:

# 4.等待新客戶端的鏈接

? ? ? ? try:

new_socket, client_addr = tcp_server_socket.accept()

# 局部變量

? ? ? ? except Exception as ret:

pass

? ? ? ? else:

new_socket.setblocking(False)

client_socket_list.append(new_socket)

for client_socketin client_socket_list:

try:

recv_data = client_socket.recv(1024).dacode('utf-8')

except Exception as ret:

pass

? ? ? ? ? ? else:

if recv_data:

# 5.為這個客戶端服務

? ? ? ? ? ? ? ? ? ? service_client(client_socket,recv_data)

else:

client_socket.close()

client_socket_list.remove(client_socket)

# 6.關閉監聽套接字

? ? tcp_server_socket.close()

if __name__ =='__main__':

main()

epoll實現http服務器

import socket

import re

import select

def service_client(new_socket):

# '''為這個客戶端返回數據'''

# 1.接收瀏覽器發送出來的請求,即HTTP請求

# GET/HTTP/1.1

#

request_lines = request.splotlines()

print ('')

print ('>>>' *30)

print (request_lines)

ret = re.match(r'[^/]+(/[^ ]*)', request_lines[0])#正則提取

file_name =''

if ret:

file_name = ret.group(1)

print ('*' *50, file_name)# !!!!!!!!網頁名字

if file_name ='/index.html'

# 2.返回HTTP格式的數據給瀏覽器

try:

f =open('.../Desktop/項目/項目.html', 'rb')

# f = open('.../Desktop/項目/項目.html' +file_name,'rb')

except:

response ='HTTP/1.1 404 NOT FOUND\r\n'

response +='\r\n'

response +='--FILE NOT FOUND---'

new_socket.send(response.encode('utf-8'))

else:

html_content = f.read()

f.close()

# 2.1 準備發送給瀏覽器的Header

response_header ='HTTP/1.1 200 OK\r\n'

response_header +='\r\n'

response_header ='Content-Length : %d\r\n'%len(response_body)

response_header +='\r\n'

response = response_header.encode('utf-8')+ response_body

#將 Response發送給瀏覽器

new_socket.send(response)

def main():

""" 用來完成整提的控制"""

# 1.創建套接字

tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

tcp_server_socket.setsockopt(socket.SOCK_STREAM, socket.SO_REUSEADDR,1)

# 2.綁定

tcp_server_socket.bind(('', 7890))

# 3.變為監聽套接字(最大鏈接書128)

tcp_server_socket.listen(128)

tcp_server_socket.setblocking(False)#將套接字變為非阻塞

#創建一個epoll對象

epl = select.epoll()

#將監聽套接字對應的FD注冊到epoll中

epl.register(tcp_server_socket.fileno(),select.EPOLLIN)

client_socket_list =list()

while True:

fd_event_list = epl.poll()#默認會阻塞,直到os檢測到數據到來,

# 通過事件通知的方式告訴程序,此時才會解阻塞

# [(fd ,event),(...)]

#參數fd:套接字對應的文件描述符

#參數event:這個fd是什么事件,例如可以調用recv 接收等

for fd,eventin fd_event_list:

# 4.等待新客戶端的鏈接

if fd == tcp_server_socket.fileno():

new_socket,client_addr = tcp_server_socket.accept()

epl.register(new_socket.fileno(),select.EPOLLIN)

elif event == select.EPOLLIN:

#判斷已經連接的客戶端是否有數據發送過來

recv_data = client_socket.recv(1024).decode('utf-8')

if recv_data:

#為這個客戶端服務

service_client(fd_event_dict[fd],recv_data)

else:

fd_event_list[fd].close()

epl.unregister(fd)

del fd_event_dict[fd]

# 6.關閉監聽套接字

tcp_server_socket.close()

if __name__ =='__main__':

main()

epoll的原理

使用了內存映射技術 mmap

epoll采用基于事件的就緒通知方式


最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。