30.4. 企业级开发进阶2.4:服务端快捷开发

前面的内容中已经介绍了TCP编程和UDP编程实现网络数据的通信和共享

我们可以看到客户端程序的编写相对容易,主需要连接服务端然后跟服务端进行数据交互就OK了。但是服务端的程序编写较为复杂,如果考虑到数据的并发处理等各种问题,就更加复杂难以操作了。

python提供了一个socketserver模块,可以用于更加快捷的构建我们需要的服务端环境

本节内容

  1. socketserver模块简介
  2. 常规模式服务端编程
  3. 并发模式服务端编程

1. socketserver模块简介

  • socketserver是什么?
    socketserver模块时python提供的内置的用于快捷开发服务端程序的一个服务器框架,通过封装大量实现的方式减少开发人员工作量的同时能快捷开发出具有较高质量的服务端程序。

  • socketserver中提供了什么?
    socketserver模块主要包含的服务器类:TCPserver、UCPserver、ThreadingTCPserver、ThreadingUDPserver、ForkingTCPserver、ForkingUDPserver
    注意:上述TCP表示TCP服务端编程需要的服务类,UDP表示UDP编程需要的服务类,包含Threading的表示多线程并发需要的服务类;包含Forking的表示多进程并发需要的服务类
    关于多线程和多进程,后面的章节中会详细介绍

  • socketserver核心操作?
    socketserver框架中,服务端的处理类主要有StreamRequestHandler(基于TCP协议的)、DatagramRequestHandler(基于UDP协议的),处理类中非常重要的一个方法headler()用来执行服务端程序中的核心操作

class mytcpserver(socketserver.StreamRequestHandler):
    def handle(self):
        # 服务器中的核心操作代码
  • socketserver中的结构?
    socketserver中主要包含两部分:服务器和处理类
    服务类就是socketserver提供了内置服务类,如TCPserver、UDPserver等等
    处理类就是我们自定义的处理类,处理类中会包含handle()方法用于业务处理

2. 服务端编程

2.1. TCP服务端编程

使用socketserver编写服务端程序如下:

import socketserver


# 自定义处理类
class myTcp(socketserver.StreamRequestHandler):
    # 定义处理方法
    def handle(self):
        # 通过client_address属性查看连接进来的服务器
        print("连接上的服务器:%s" % str(self.client_address))
        while True:
            # 接收客户端发送的数据
            msg = self.request.recv(1024)
            if not msg:
                break;
            print("客户端发过来消息:%s" % msg.decode("UTF-8"))
            # 给客户端返回接收信息
            self.request.sendall("已经成功接收您发送的消息".encode("UTF-8"))


# 程序从主线程直接运行
if __name__ == "__main__":
    # 创建服务端对象,指定处理类,并监听8888端口
    server = socketserver.TCPServer(('', 8888), myTcp)
    # 启动服务端程序
    server.serve_forever()

此时,我们使用常规模式开发客户端程序如下:

import socket

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

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

while True:
    msg = input("请输入要发送的内容:")
    if not msg or msg == "exit":
        break
    sc.sendall(msg.encode("UTF-8"))
    msg = sc.recv(1024)
    print("服务器回应:" + msg.decode("UTF-8"))

sc.close()

运行上述程序,就可以和之前一样,实现客户端和服务端之间的数据通信了;
和以前不一样的时,此时我们如果关闭了客户端,服务端程序还是在运行的,如果重新启动客户端,客户端又会接入服务端,重新建立连接并通信。

2.2. UDP服务端编程

使用socketserver的UDPServer服务类和DatagramRequestHandler处理类进行服务端的编程处理如下:

import socketserver

# 创建自定义处理类
class myUdp(socketserver.DatagramRequestHandler):
    # 创建自定义处理方法
    def handle(self):
        # 打印连接进来的客户端信息
        print("连接到服务器的主机:" + str(self.client_address))

        # 收发消息
        while True:
            # 接收客户端发送的消息
            # msg = self.request.recv()
            msg = self.rfile.readline()
            if not msg:
                break
            print("接受到客户端发送的消息:%s" % msg.decode("UTF-8"))
            # 发送消息
            self.wfile.write("接收到您发送的消息".encode("UTF-8"))
# 主模块线程中运行程序
if __name__ == "__main__":
    # 创建UDP服务器对象
    server = socketserver.UDPServer(("", 8989), myUdp)
    # 启动UDP服务器
    server.serve_forever()

使用常规的方式开发UDP客户端代码如下:

import socket

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

while True:
    data = input("请输入要发送的消息:")
    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("服务器返回消息:" + data.decode("UTF-8"))

sc.close()

此时启动服务端程序,启动客户端程序,就可以实现服务端和客户端时间的数据通信了。
另外:你可以试试启动多个客户端程序看看,有彩蛋哦。

3. 服务端并发

关于使用多线程或者多进程并发的方式也是比较简单的,参考代码如下,如果有兴趣的话,等学习完并发编程之后,可以再回过头看看ThreadingTCPServer和ForkingTCPServer这样多线程和多进程并发的操作哦
服务端参考代码:

import socketserver

class myTcp(socketserver.StreamRequestHandler):
    def handle(self):
        while True:
            data = self.request.recv(1024)
            print("接收到数据:" + data.decode("UTF-8"))
            self.request.sendall("数据已经接收成功".encode("UTF-8"))

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

客户端还是常规的客户端

import socket

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

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

while True:
    msg = input("请输入要发送的内容:")
    if not msg or msg == "exit":
        break
    sc.sendall(msg.encode("UTF-8"))
    msg = sc.recv(1024)
    print("服务器回应:" + msg.decode("UTF-8"))

sc.close()

大牧莫邪.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,967评论 19 139
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,370评论 11 349
  • 我愿意 与你一起 不分离 什么时候 我对你的爱 变成了 怀疑 赞美的歌 出口却是 怨言 看不见了你面 我是孤儿
    金火丁阅读 106评论 0 0
  • 有聊的来的朋友真好,他能理解你的笑点,你能接受他的玩笑。
    我很酷所以我不吃青菜阅读 267评论 0 0
  • 作者:沈淋 导师:刘艳 袁浩 郑鹏 这是一幅自我介绍的思维导图,第一个主干是我的家乡成都,熊猫是成都的标志,一提到...
    梅女士51阅读 474评论 1 1