蜜桃精品成人影片_99热在线精品免费_日韩亚洲中字无码一区二区三区_亚洲中文字幕久久精品无码一区

您現(xiàn)在所在的位置:首頁(yè) >學(xué)習(xí)資源 > Python全棧+人工智能入門(mén)教材 > Python基礎(chǔ)入門(mén)教程30.4:企業(yè)級(jí)開(kāi)發(fā)進(jìn)階 服務(wù)端快捷開(kāi)發(fā)

Python基礎(chǔ)入門(mén)教程30.4:企業(yè)級(jí)開(kāi)發(fā)進(jìn)階 服務(wù)端快捷開(kāi)發(fā)

來(lái)源:奇酷教育 發(fā)表于:

前面的內(nèi)容中已經(jīng)介紹了TCP編程和UDP編程實(shí)現(xiàn)網(wǎng)絡(luò)數(shù)據(jù)的通信和共享我們可以看到客戶(hù)端程序的編寫(xiě)相對(duì)容易,主需要連接服務(wù)端然后跟服務(wù)端進(jìn)

前面的內(nèi)容中已經(jīng)介紹了TCP編程和UDP編程實(shí)現(xiàn)網(wǎng)絡(luò)數(shù)據(jù)的通信和共享

我們可以看到客戶(hù)端程序的編寫(xiě)相對(duì)容易,主需要連接服務(wù)端然后跟服務(wù)端進(jìn)行數(shù)據(jù)交互就OK了。但是服務(wù)端的程序編寫(xiě)較為復(fù)雜,如果考慮到數(shù)據(jù)的并發(fā)處理等各種問(wèn)題,就更加復(fù)雜難以操作了。

python提供了一個(gè)socketserver模塊,可以用于更加快捷的構(gòu)建我們需要的服務(wù)端環(huán)境

本節(jié)內(nèi)容

  1. socketserver模塊簡(jiǎn)介
  2. 常規(guī)模式服務(wù)端編程
  3. 并發(fā)模式服務(wù)端編程

1. socketserver模塊簡(jiǎn)介

  • socketserver是什么?
    socketserver模塊時(shí)python提供的內(nèi)置的用于快捷開(kāi)發(fā)服務(wù)端程序的一個(gè)服務(wù)器框架,通過(guò)封裝大量實(shí)現(xiàn)的方式減少開(kāi)發(fā)人員工作量的同時(shí)能快捷開(kāi)發(fā)出具有較高質(zhì)量的服務(wù)端程序。

  • socketserver中提供了什么?
    socketserver模塊主要包含的服務(wù)器類(lèi):TCPserver、UCPserver、ThreadingTCPserver、ThreadingUDPserver、ForkingTCPserver、ForkingUDPserver
    注意:上述TCP表示TCP服務(wù)端編程需要的服務(wù)類(lèi),UDP表示UDP編程需要的服務(wù)類(lèi),包含Threading的表示多線程并發(fā)需要的服務(wù)類(lèi);包含F(xiàn)orking的表示多進(jìn)程并發(fā)需要的服務(wù)類(lèi)
    關(guān)于多線程和多進(jìn)程,后面的章節(jié)中會(huì)詳細(xì)介紹

  • socketserver核心操作?
    socketserver框架中,服務(wù)端的處理類(lèi)主要有StreamRequestHandler(基于TCP協(xié)議的)、DatagramRequestHandler(基于UDP協(xié)議的),處理類(lèi)中非常重要的一個(gè)方法headler()用來(lái)執(zhí)行服務(wù)端程序中的核心操作

    class mytcpserver(socketserver.StreamRequestHandler):
      def handle(self):
          # 服務(wù)器中的核心操作代碼
  • socketserver中的結(jié)構(gòu)?
    socketserver中主要包含兩部分:服務(wù)器和處理類(lèi)
    服務(wù)類(lèi)就是socketserver提供了內(nèi)置服務(wù)類(lèi),如TCPserver、UDPserver等等
    處理類(lèi)就是我們自定義的處理類(lèi),處理類(lèi)中會(huì)包含handle()方法用于業(yè)務(wù)處理

2. 服務(wù)端編程

2.1. TCP服務(wù)端編程

使用socketserver編寫(xiě)服務(wù)端程序如下:

import socketserver


# 自定義處理類(lèi)
class myTcp(socketserver.StreamRequestHandler):
    # 定義處理方法
    def handle(self):
        # 通過(guò)client_address屬性查看連接進(jìn)來(lái)的服務(wù)器
        print("連接上的服務(wù)器:%s" % str(self.client_address))
        while True:
            # 接收客戶(hù)端發(fā)送的數(shù)據(jù)
            msg = self.request.recv(1024)
            if not msg:
                break;
            print("客戶(hù)端發(fā)過(guò)來(lái)消息:%s" % msg.decode("UTF-8"))
            # 給客戶(hù)端返回接收信息
            self.request.sendall("已經(jīng)成功接收您發(fā)送的消息".encode("UTF-8"))


# 程序從主線程直接運(yùn)行
if __name__ == "__main__":
    # 創(chuàng)建服務(wù)端對(duì)象,指定處理類(lèi),并監(jiān)聽(tīng)8888端口
    server = socketserver.TCPServer(('', 8888), myTcp)
    # 啟動(dòng)服務(wù)端程序
    server.serve_forever()

此時(shí),我們使用常規(guī)模式開(kāi)發(fā)客戶(hù)端程序如下:

import socket

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

sc.connect(("192.168.10.108", 8888))

while True:
    msg = input("請(qǐng)輸入要發(fā)送的內(nèi)容:")
    if not msg or msg == "exit":
        break
    sc.sendall(msg.encode("UTF-8"))
    msg = sc.recv(1024)
    print("服務(wù)器回應(yīng):" + msg.decode("UTF-8"))

sc.close()

運(yùn)行上述程序,就可以和之前一樣,實(shí)現(xiàn)客戶(hù)端和服務(wù)端之間的數(shù)據(jù)通信了;
和以前不一樣的時(shí),此時(shí)我們?nèi)绻P(guān)閉了客戶(hù)端,服務(wù)端程序還是在運(yùn)行的,如果重新啟動(dòng)客戶(hù)端,客戶(hù)端又會(huì)接入服務(wù)端,重新建立連接并通信。

2.2. UDP服務(wù)端編程

使用socketserver的UDPServer服務(wù)類(lèi)和DatagramRequestHandler處理類(lèi)進(jìn)行服務(wù)端的編程處理如下:

import socketserver

# 創(chuàng)建自定義處理類(lèi)
class myUdp(socketserver.DatagramRequestHandler):
    # 創(chuàng)建自定義處理方法
    def handle(self):
        # 打印連接進(jìn)來(lái)的客戶(hù)端信息
        print("連接到服務(wù)器的主機(jī):" + str(self.client_address))

        # 收發(fā)消息
        while True:
            # 接收客戶(hù)端發(fā)送的消息
            # msg = self.request.recv()
            msg = self.rfile.readline()
            if not msg:
                break
            print("接受到客戶(hù)端發(fā)送的消息:%s" % msg.decode("UTF-8"))
            # 發(fā)送消息
            self.wfile.write("接收到您發(fā)送的消息".encode("UTF-8"))
# 主模塊線程中運(yùn)行程序
if __name__ == "__main__":
    # 創(chuàng)建UDP服務(wù)器對(duì)象
    server = socketserver.UDPServer(("", 8989), myUdp)
    # 啟動(dòng)UDP服務(wù)器
    server.serve_forever()

使用常規(guī)的方式開(kāi)發(fā)UDP客戶(hù)端代碼如下:

import socket

sc = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)

while True:
    data = input("請(qǐng)輸入要發(fā)送的消息:")
    if not data or data == "exit":
        break

    sc.sendto(data.encode("UTF-8"), ("192.168.10.108", 8989))

    data,recv = sc.recvfrom(1024)
    if not data:
        break
    print("服務(wù)器返回消息:" + data.decode("UTF-8"))

sc.close()

此時(shí)啟動(dòng)服務(wù)端程序,啟動(dòng)客戶(hù)端程序,就可以實(shí)現(xiàn)服務(wù)端和客戶(hù)端時(shí)間的數(shù)據(jù)通信了。
另外:你可以試試啟動(dòng)多個(gè)客戶(hù)端程序看看,有彩蛋哦。

3. 服務(wù)端并發(fā)

關(guān)于使用多線程或者多進(jìn)程并發(fā)的方式也是比較簡(jiǎn)單的,參考代碼如下,如果有興趣的話,等學(xué)習(xí)完并發(fā)編程之后,可以再回過(guò)頭看看ThreadingTCPServer和ForkingTCPServer這樣多線程和多進(jìn)程并發(fā)的操作哦
服務(wù)端參考代碼:

import socketserver

class myTcp(socketserver.StreamRequestHandler):
    def handle(self):
        while True:
            data = self.request.recv(1024)
            print("接收到數(shù)據(jù):" + data.decode("UTF-8"))
            self.request.sendall("數(shù)據(jù)已經(jīng)接收成功".encode("UTF-8"))

if __name__ == "__main__":
    server = socketserver.ThreadingTCPServer(("", 9000), myTcp)
    server.serve_forever()

客戶(hù)端還是常規(guī)的客戶(hù)端

import socket

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

sc.connect(("192.168.10.108", 9000))

while True:
    msg = input("請(qǐng)輸入要發(fā)送的內(nèi)容:")
    if not msg or msg == "exit":
        break
    sc.sendall(msg.encode("UTF-8"))
    msg = sc.recv(1024)
    print("服務(wù)器回應(yīng):" + msg.decode("UTF-8"))

sc.close()

>>>更多VR/AR入門(mén)教程:VR入門(mén)
 
昌宁县| 巴林右旗| 崇文区| 杭锦旗| 都兰县| 蓝山县| 汝城县| 三台县| 灌云县| 盐源县| 剑河县| 九江县| 岱山县| 神木县| 汉阴县| 梅河口市| 阳泉市| 焦作市| 盘山县| 莲花县| 青冈县| 靖安县| 宜兴市| 五大连池市| 德江县| 郓城县| 沅陵县| 德保县| 衢州市| 屏东市| 泽州县| 昌江| 甘德县| 通山县| 福贡县| 石渠县| 荣成市| 临江市| 邵东县| 读书| 望谟县|