Python實現HTTP服務器(四)線程、進程、協程實現多任務

承接上文:https://blog.csdn.net/qq_32426313/article/details/104192330

1.線程、進程實現多任務HTTP服務器

import re
import socket
import multiprocessing
import threading

def service_client(new_socket):
    """爲這個客戶端返回數據"""

    #  接收瀏覽器發送過來的請求,即HTTP請求
    request=new_socket.recv(1024)
    request=request.decode("utf-8")  #  解碼
    request_lines=request.splitlines()  #  按照行('\r', '\r\n', \n')分隔,返回一個包含各行作爲元素的列表
    print(request_lines)

    #  GET /index.html HTTP/1.1
    #  [^/]表示除了/都可以
    file_name=""
    ret=re.match(r"[^/]+(/[^ ]*)",request_lines[0])
    if ret:
        file_name=ret.group(1)
        if file_name=="/":
            file_name="/index.html"


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

    try:
        #  準備發送的body,打開HTML文件
        f=open("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()
        #  準備發送的header
        response="HTTP/1.1 200 OK\r\n"
        response+="\r\n"  #  header與body之間必須隔一行
        #  發送header
        new_socket.send(response.encode("utf-8"))
        #  發送HTML
        new_socket.send(html_content)


    new_socket.close()

def main():
    tcp_sever_socket=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
    #  服務器先關閉,保證重新開啓不佔用端口
    tcp_sever_socket.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1)
    tcp_sever_socket.bind(("",7890))
    tcp_sever_socket.listen(128)
    while True:
        #  等待新客戶端的鏈接
        new_socket,client_addr=tcp_sever_socket.accept()

        #p=multiprocessing.Process(target=service_client,args=(new_socket,))
        p=threading.Thread(target=service_client,args=(new_socket,))
        p.start()
        #  子進程關閉一次主進程還需關閉一次,Linux下文件硬鏈接的原因,兩個名字指向同一個文件,因此要關閉兩次
        #new_socket.close()
    #  關閉監聽套接字
    tcp_sever_socket.close()
if __name__ == '__main__':
    main()

2.gevent協程實現多任務HTTP服務器

import re
import socket
import gevent
from gevent import monkey

monkey.patch_all()
def service_client(new_socket):
    """爲這個客戶端返回數據"""

    #  接收瀏覽器發送過來的請求,即HTTP請求
    request=new_socket.recv(1024)
    request=request.decode("utf-8")  #  解碼
    request_lines=request.splitlines()  #  按照行('\r', '\r\n', \n')分隔,返回一個包含各行作爲元素的列表
    print(request_lines)

    #  GET /index.html HTTP/1.1
    #  [^/]表示除了/都可以
    file_name=""
    ret=re.match(r"[^/]+(/[^ ]*)",request_lines[0])
    if ret:
        file_name=ret.group(1)
        if file_name=="/":
            file_name="/index.html"


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

    try:
        #  準備發送的body,打開HTML文件
        f=open("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()
        #  準備發送的header
        response="HTTP/1.1 200 OK\r\n"
        response+="\r\n"  #  header與body之間必須隔一行
        #  發送header
        new_socket.send(response.encode("utf-8"))
        #  發送HTML
        new_socket.send(html_content)


    new_socket.close()

def main():
    tcp_sever_socket=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
    #  服務器先關閉,保證重新開啓不佔用端口
    tcp_sever_socket.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1)
    tcp_sever_socket.bind(("",7890))
    tcp_sever_socket.listen(128)
    while True:
        #  等待新客戶端的鏈接
        new_socket,client_addr=tcp_sever_socket.accept()

        gevent.spawn(service_client,new_socket)
    #  關閉監聽套接字
    tcp_sever_socket.close()
if __name__ == '__main__':
    main()

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章